diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..e53afba --- /dev/null +++ b/.gitignore @@ -0,0 +1,20 @@ +# Node +node_modules/ + +# Build output +dist/ + +# Rust / Tauri +src-tauri/target/ +src-tauri/gen/ + +# Local dev config (machine-specific paths) +src-tauri/.cargo/config.toml + +# Editor +.vscode/ +.idea/ + +# Env +.env +.env.local diff --git a/docs/superpowers/plans/2026-04-13-orchai-phase1-foundation.md b/docs/superpowers/plans/2026-04-13-orchai-phase1-foundation.md index 4bbe356..2f6ab16 100644 --- a/docs/superpowers/plans/2026-04-13-orchai-phase1-foundation.md +++ b/docs/superpowers/plans/2026-04-13-orchai-phase1-foundation.md @@ -1,6 +1,6 @@ # Orchai Phase 1: Foundation Implementation Plan -> **For agentic workers:** REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (`- [ ]`) syntax for tracking. +> **For agentic workers:** REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (`- [x]`) syntax for tracking. **Goal:** Get a working Tauri 2 desktop app with SQLite storage and full Project CRUD (create from local path or clone URL, list, edit, delete) with a React UI. @@ -78,7 +78,7 @@ orchai/ - Create: entire project scaffold via CLI - Preserve: `docs/`, `.git/` -- [ ] **Step 1: Save existing repo contents** +- [x] **Step 1: Save existing repo contents** ```bash cd /home/leclere/Projets @@ -86,7 +86,7 @@ cp -r orchai/docs /tmp/orchai-docs-backup cp -r orchai/.git /tmp/orchai-git-backup ``` -- [ ] **Step 2: Scaffold Tauri 2 project** +- [x] **Step 2: Scaffold Tauri 2 project** ```bash cd /home/leclere/Projets @@ -102,7 +102,7 @@ When prompted, select: - UI template: `React` - UI flavor: `TypeScript` -- [ ] **Step 3: Restore repo history and docs** +- [x] **Step 3: Restore repo history and docs** ```bash cd /home/leclere/Projets/orchai @@ -112,7 +112,7 @@ cp -r /tmp/orchai-docs-backup docs rm -rf /tmp/orchai-docs-backup /tmp/orchai-git-backup ``` -- [ ] **Step 4: Install dependencies and verify** +- [x] **Step 4: Install dependencies and verify** ```bash cd /home/leclere/Projets/orchai @@ -123,7 +123,7 @@ cd .. Expected: build succeeds with no errors. -- [ ] **Step 5: Verify dev server starts** +- [x] **Step 5: Verify dev server starts** ```bash npm run tauri dev @@ -131,7 +131,7 @@ npm run tauri dev Expected: Tauri window opens with the default React starter page. Close it after verifying. -- [ ] **Step 6: Commit** +- [x] **Step 6: Commit** ```bash git add -A @@ -147,14 +147,14 @@ git commit -m "scaffold: Tauri 2 + React + TypeScript via create-tauri-app" - Modify: `src/index.css`, `package.json`, `src-tauri/tauri.conf.json` - Delete: `src/App.css` -- [ ] **Step 1: Install Tailwind** +- [x] **Step 1: Install Tailwind** ```bash cd /home/leclere/Projets/orchai npm install -D tailwindcss @tailwindcss/vite ``` -- [ ] **Step 2: Add Tailwind to Vite config** +- [x] **Step 2: Add Tailwind to Vite config** Replace the contents of `vite.config.ts`: @@ -186,7 +186,7 @@ export default defineConfig(async () => ({ })); ``` -- [ ] **Step 3: Replace index.css with Tailwind directives** +- [x] **Step 3: Replace index.css with Tailwind directives** Replace the contents of `src/index.css`: @@ -194,7 +194,7 @@ Replace the contents of `src/index.css`: @import "tailwindcss"; ``` -- [ ] **Step 4: Delete App.css and clean up App.tsx** +- [x] **Step 4: Delete App.css and clean up App.tsx** Delete `src/App.css`. @@ -212,7 +212,7 @@ function App() { export default App; ``` -- [ ] **Step 5: Update Tauri config** +- [x] **Step 5: Update Tauri config** In `src-tauri/tauri.conf.json`, update the `app` section: @@ -244,7 +244,7 @@ In `src-tauri/tauri.conf.json`, update the `app` section: } ``` -- [ ] **Step 6: Verify Tailwind works** +- [x] **Step 6: Verify Tailwind works** ```bash npm run tauri dev @@ -252,7 +252,7 @@ npm run tauri dev Expected: window opens, "Orchai" heading rendered with Tailwind styling (bold, large). Close after verifying. -- [ ] **Step 7: Commit** +- [x] **Step 7: Commit** ```bash git add -A @@ -270,7 +270,7 @@ git commit -m "configure: Tailwind CSS + app metadata" - Create: `src-tauri/src/error.rs` - Modify: `src-tauri/src/lib.rs` -- [ ] **Step 1: Write the failing test for db initialization** +- [x] **Step 1: Write the failing test for db initialization** Add dependencies to `src-tauri/Cargo.toml` under `[dependencies]`: @@ -344,7 +344,7 @@ mod tests { } ``` -- [ ] **Step 2: Run tests to verify they fail** +- [x] **Step 2: Run tests to verify they fail** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -353,7 +353,7 @@ cargo test db::tests Expected: 3 failures with `not yet implemented`. -- [ ] **Step 3: Create migration SQL** +- [x] **Step 3: Create migration SQL** Create `src-tauri/migrations/001_init.sql`: @@ -421,7 +421,7 @@ CREATE TABLE IF NOT EXISTS notifications ( ); ``` -- [ ] **Step 4: Implement db::init and db::init_in_memory** +- [x] **Step 4: Implement db::init and db::init_in_memory** Replace the `todo!()` implementations in `src-tauri/src/db.rs`: @@ -463,7 +463,7 @@ fn migrate(conn: &Connection) -> Result<()> { } ``` -- [ ] **Step 5: Run tests to verify they pass** +- [x] **Step 5: Run tests to verify they pass** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -489,7 +489,7 @@ Replace the WAL test assertion: Re-run tests. Expected: 3 pass. -- [ ] **Step 6: Create error type** +- [x] **Step 6: Create error type** Create `src-tauri/src/error.rs`: @@ -530,7 +530,7 @@ impl std::fmt::Display for AppError { } ``` -- [ ] **Step 7: Wire up db module in lib.rs** +- [x] **Step 7: Wire up db module in lib.rs** Replace `src-tauri/src/lib.rs`: @@ -562,7 +562,7 @@ pub fn run() { } ``` -- [ ] **Step 8: Verify it compiles and runs** +- [x] **Step 8: Verify it compiles and runs** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -571,7 +571,7 @@ cargo build Expected: compiles with no errors. -- [ ] **Step 9: Commit** +- [x] **Step 9: Commit** ```bash git add -A @@ -587,7 +587,7 @@ git commit -m "feat: SQLite database with migration system and full schema" - Create: `src-tauri/src/models/project.rs` - Modify: `src-tauri/src/lib.rs` (add `mod models`) -- [ ] **Step 1: Write failing tests for Project CRUD** +- [x] **Step 1: Write failing tests for Project CRUD** Create `src-tauri/src/models/mod.rs`: @@ -740,7 +740,7 @@ mod tests { Add `mod models;` to `src-tauri/src/lib.rs` (after `mod error;`). -- [ ] **Step 2: Run tests to verify they fail** +- [x] **Step 2: Run tests to verify they fail** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -749,7 +749,7 @@ cargo test models::project::tests Expected: 8 failures with `not yet implemented`. -- [ ] **Step 3: Implement Project CRUD** +- [x] **Step 3: Implement Project CRUD** Replace the `todo!()` implementations in `src-tauri/src/models/project.rs`: @@ -833,7 +833,7 @@ Add the missing imports at the top of the file: use chrono; ``` -- [ ] **Step 4: Run tests to verify they pass** +- [x] **Step 4: Run tests to verify they pass** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -842,7 +842,7 @@ cargo test models::project::tests Expected: 8 tests pass. -- [ ] **Step 5: Commit** +- [x] **Step 5: Commit** ```bash git add -A @@ -859,7 +859,7 @@ git commit -m "feat: Project model with CRUD operations and tests" - Modify: `src-tauri/src/lib.rs` - Modify: `src-tauri/Cargo.toml` (add tauri-plugin-dialog) -- [ ] **Step 1: Add dialog plugin dependency** +- [x] **Step 1: Add dialog plugin dependency** Add to `src-tauri/Cargo.toml` under `[dependencies]`: @@ -873,7 +873,7 @@ Add to the `capabilities/default.json` permissions array: "dialog:default" ``` -- [ ] **Step 2: Create commands module** +- [x] **Step 2: Create commands module** Create `src-tauri/src/commands/mod.rs`: @@ -966,7 +966,7 @@ pub fn delete_project(state: State<'_, AppState>, id: String) -> Result<(), AppE } ``` -- [ ] **Step 3: Add `dirs` dependency** +- [x] **Step 3: Add `dirs` dependency** Add to `src-tauri/Cargo.toml` under `[dependencies]`: @@ -974,7 +974,7 @@ Add to `src-tauri/Cargo.toml` under `[dependencies]`: dirs = "5" ``` -- [ ] **Step 4: Wire up commands in lib.rs** +- [x] **Step 4: Wire up commands in lib.rs** Replace `src-tauri/src/lib.rs`: @@ -1016,7 +1016,7 @@ pub fn run() { } ``` -- [ ] **Step 5: Verify it compiles** +- [x] **Step 5: Verify it compiles** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -1025,7 +1025,7 @@ cargo build Expected: compiles with no errors. -- [ ] **Step 6: Commit** +- [x] **Step 6: Commit** ```bash git add -A @@ -1040,7 +1040,7 @@ git commit -m "feat: Tauri commands for project CRUD with git clone support" - Create: `src/lib/types.ts` - Create: `src/lib/api.ts` -- [ ] **Step 1: Install frontend dependencies** +- [x] **Step 1: Install frontend dependencies** ```bash cd /home/leclere/Projets/orchai @@ -1048,7 +1048,7 @@ npm install react-router-dom npm install @tauri-apps/plugin-dialog ``` -- [ ] **Step 2: Create TypeScript types** +- [x] **Step 2: Create TypeScript types** Create `src/lib/types.ts`: @@ -1063,7 +1063,7 @@ export interface Project { } ``` -- [ ] **Step 3: Create API wrapper** +- [x] **Step 3: Create API wrapper** Create `src/lib/api.ts`: @@ -1104,7 +1104,7 @@ export async function deleteProject(id: string): Promise { } ``` -- [ ] **Step 4: Commit** +- [x] **Step 4: Commit** ```bash git add -A @@ -1120,7 +1120,7 @@ git commit -m "feat: TypeScript types and Tauri API wrappers for project CRUD" - Create: `src/components/layout/AppLayout.tsx` - Create: `src/components/layout/Sidebar.tsx` -- [ ] **Step 1: Create Sidebar component** +- [x] **Step 1: Create Sidebar component** Create directory structure: @@ -1196,7 +1196,7 @@ export default function Sidebar() { } ``` -- [ ] **Step 2: Create AppLayout component** +- [x] **Step 2: Create AppLayout component** Create `src/components/layout/AppLayout.tsx`: @@ -1216,7 +1216,7 @@ export default function AppLayout() { } ``` -- [ ] **Step 3: Set up router in App.tsx** +- [x] **Step 3: Set up router in App.tsx** Replace `src/App.tsx`: @@ -1251,7 +1251,7 @@ function App() { export default App; ``` -- [ ] **Step 4: Verify the shell renders** +- [x] **Step 4: Verify the shell renders** ```bash npm run tauri dev @@ -1259,7 +1259,7 @@ npm run tauri dev Expected: window opens with dark sidebar on the left showing "Orchai" header, "Projects" section with "No projects yet" message, and a "+" button. Main area shows "Select a project or create a new one". Close after verifying. -- [ ] **Step 5: Commit** +- [x] **Step 5: Commit** ```bash git add -A @@ -1275,7 +1275,7 @@ git commit -m "feat: React app shell with router, sidebar layout" - Create: `src/components/projects/ProjectDashboard.tsx` - Modify: `src/App.tsx` -- [ ] **Step 1: Create ProjectForm component** +- [x] **Step 1: Create ProjectForm component** Create `src/components/projects/ProjectForm.tsx`: @@ -1456,7 +1456,7 @@ export default function ProjectForm() { } ``` -- [ ] **Step 2: Create ProjectDashboard placeholder** +- [x] **Step 2: Create ProjectDashboard placeholder** Create `src/components/projects/ProjectDashboard.tsx`: @@ -1539,7 +1539,7 @@ export default function ProjectDashboard() { } ``` -- [ ] **Step 3: Wire up routes in App.tsx** +- [x] **Step 3: Wire up routes in App.tsx** Replace `src/App.tsx`: @@ -1576,7 +1576,7 @@ function App() { export default App; ``` -- [ ] **Step 4: Verify the full flow in the browser** +- [x] **Step 4: Verify the full flow in the browser** ```bash npm run tauri dev @@ -1590,7 +1590,7 @@ Test the following: 5. Click "Edit" -- form pre-fills with project data 6. Click "Delete" -- project removed from sidebar -- [ ] **Step 5: Commit** +- [x] **Step 5: Commit** ```bash git add -A @@ -1605,7 +1605,7 @@ git commit -m "feat: project create/edit/delete UI with folder picker and git cl - Verify all tests pass - Clean up any scaffold files not needed -- [ ] **Step 1: Run all Rust tests** +- [x] **Step 1: Run all Rust tests** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -1614,7 +1614,7 @@ cargo test Expected: all tests pass (8 model tests + 3 db tests = 11 tests). -- [ ] **Step 2: Run Rust clippy** +- [x] **Step 2: Run Rust clippy** ```bash cd /home/leclere/Projets/orchai/src-tauri @@ -1623,7 +1623,7 @@ cargo clippy -- -D warnings Expected: no warnings. If there are warnings, fix them. -- [ ] **Step 3: Verify frontend builds** +- [x] **Step 3: Verify frontend builds** ```bash cd /home/leclere/Projets/orchai @@ -1632,13 +1632,13 @@ npm run build Expected: Vite build succeeds. -- [ ] **Step 4: Clean up scaffold files** +- [x] **Step 4: Clean up scaffold files** Remove any remaining scaffold assets that are not needed: - `src/assets/react.svg` (if still present) - Any other default scaffold content -- [ ] **Step 5: Final integration test** +- [x] **Step 5: Final integration test** ```bash npm run tauri dev @@ -1652,7 +1652,7 @@ Test the complete flow one more time: 5. Delete the project 6. Verify sidebar is empty again -- [ ] **Step 6: Commit** +- [x] **Step 6: Commit** ```bash git add -A diff --git a/docs/superpowers/plans/2026-04-13-orchai-phase2-tuleap-integration.md b/docs/superpowers/plans/2026-04-13-orchai-phase2-tuleap-integration.md new file mode 100644 index 0000000..f28289d --- /dev/null +++ b/docs/superpowers/plans/2026-04-13-orchai-phase2-tuleap-integration.md @@ -0,0 +1,3378 @@ +# Orchai Phase 2: Tuleap Integration Implementation Plan + +> **For agentic workers:** REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (`- [x]`) syntax for tracking. + +**Goal:** Connect Orchai to Tuleap -- store credentials securely, configure watched trackers with AND/OR filters, poll for new tickets on a timer, and detect new artifacts. + +**Architecture:** Adds a `services/` layer (crypto, Tuleap HTTP client, filter engine, background poller) to the existing Tauri backend. AppState gains `Arc>` for shared DB access, an encryption key for credentials, and a shared reqwest client. The poller runs as a tokio background task spawned at app startup. Frontend gains a Settings page for credentials, a tracker config form with a visual filter builder, and an updated project dashboard showing tracked trackers and recent tickets. + +**Tech Stack:** reqwest (HTTP), tokio (async runtime / timers), aes-gcm + rand + base64 (credential encryption), serde_json (artifact parsing) + +--- + +## Phasing Context + +This is Plan 2 of 4: +- **Plan 1 (done):** Foundation -- Tauri scaffold, SQLite, Project Manager +- **Plan 2 (this):** Tuleap Integration -- credentials, API client, poller, filter engine, tracker config +- **Plan 3:** Agent Pipeline -- orchestrator, worktree manager, ticket processing, results UI +- **Plan 4:** Notifications + Polish -- notifier, system notifications, dashboard + +--- + +## File Structure + +``` +src-tauri/ + migrations/ + 001_init.sql # existing (unchanged) + 002_add_last_polled.sql # create: add last_polled_at + enabled to watched_trackers + src/ + lib.rs # modify: Arc>, add services mod, poller startup + db.rs # modify: add migration 002 + error.rs # modify: add From + models/ + mod.rs # modify: add credential, tracker, ticket + project.rs # existing (unchanged) + credential.rs # create: TuleapCredentials CRUD + tracker.rs # create: WatchedTracker CRUD + ticket.rs # create: ProcessedTicket insert/query + commands/ + mod.rs # modify: add credential, tracker, poller + project.rs # existing (unchanged) + credential.rs # create: set/get/delete/test credentials + tracker.rs # create: tracker CRUD + get_tracker_fields + poller.rs # create: manual_poll, get_queue_status, toggle + services/ + mod.rs # create: re-exports + crypto.rs # create: key file management, AES-GCM encrypt/decrypt + tuleap_client.rs # create: HTTP client (get artifacts, tracker info, test) + filter_engine.rs # create: AND/OR filter evaluation on artifact JSON + poller.rs # create: background polling loop + +src/ + lib/ + types.ts # modify: add Credential, Tracker, Filter, Ticket types + api.ts # modify: add all new Tauri command wrappers + components/ + settings/ + SettingsPage.tsx # create: credentials management + trackers/ + TrackerConfig.tsx # create: add/edit tracker form + FilterBuilder.tsx # create: visual AND/OR filter builder + TrackerList.tsx # create: list of trackers for a project + projects/ + ProjectDashboard.tsx # modify: add tracker section + recent tickets + App.tsx # modify: add /settings and /projects/:id/trackers routes +``` + +--- + +### Task 1: Add Phase 2 dependencies + migration 002 + update AppState + +**Files:** +- Modify: `src-tauri/Cargo.toml` +- Create: `src-tauri/migrations/002_add_last_polled.sql` +- Modify: `src-tauri/src/db.rs` +- Modify: `src-tauri/src/error.rs` +- Modify: `src-tauri/src/lib.rs` + +- [x] **Step 1: Add dependencies to Cargo.toml** + +Add under `[dependencies]` in `src-tauri/Cargo.toml`: + +```toml +reqwest = { version = "0.12", features = ["json"] } +tokio = { version = "1", features = ["time", "sync", "macros"] } +aes-gcm = "0.10" +rand = "0.8" +base64 = "0.22" +``` + +- [x] **Step 2: Create migration 002** + +Create `src-tauri/migrations/002_add_last_polled.sql`: + +```sql +ALTER TABLE watched_trackers ADD COLUMN last_polled_at TEXT; +ALTER TABLE watched_trackers ADD COLUMN enabled INTEGER NOT NULL DEFAULT 1; +``` + +- [x] **Step 3: Update db.rs to run migration 002** + +In `src-tauri/src/db.rs`, add the new migration constant and update the `migrate` function: + +```rust +use rusqlite::{Connection, Result}; +use std::path::Path; + +const MIGRATION_001: &str = include_str!("../migrations/001_init.sql"); +const MIGRATION_002: &str = include_str!("../migrations/002_add_last_polled.sql"); + +pub fn init(db_path: &Path) -> Result { + let conn = Connection::open(db_path)?; + configure(&conn)?; + migrate(&conn)?; + Ok(conn) +} + +#[cfg(test)] +pub fn init_in_memory() -> Result { + let conn = Connection::open_in_memory()?; + configure(&conn)?; + migrate(&conn)?; + Ok(conn) +} + +fn configure(conn: &Connection) -> Result<()> { + conn.pragma_update(None, "journal_mode", "wal")?; + conn.pragma_update(None, "foreign_keys", "ON")?; + Ok(()) +} + +fn migrate(conn: &Connection) -> Result<()> { + let version: i32 = conn.pragma_query_value(None, "user_version", |row| row.get(0))?; + + if version < 1 { + conn.execute_batch(MIGRATION_001)?; + conn.pragma_update(None, "user_version", 1)?; + } + if version < 2 { + conn.execute_batch(MIGRATION_002)?; + conn.pragma_update(None, "user_version", 2)?; + } + + Ok(()) +} +``` + +Update the test `test_migration_is_idempotent` to check for version 2: + +```rust + #[test] + fn test_migration_is_idempotent() { + let conn = init_in_memory().expect("should initialize"); + let version: i32 = conn + .pragma_query_value(None, "user_version", |row| row.get(0)) + .unwrap(); + assert_eq!(version, 2); + } +``` + +- [x] **Step 4: Add From to error.rs** + +Add to `src-tauri/src/error.rs`: + +```rust +impl From for AppError { + fn from(e: reqwest::Error) -> Self { + AppError { + message: e.to_string(), + } + } +} +``` + +- [x] **Step 5: Update AppState to use Arc> and add new fields** + +Replace `src-tauri/src/lib.rs`: + +```rust +mod commands; +mod db; +mod error; +mod models; + +use std::sync::{Arc, Mutex}; +use tauri::Manager; + +pub struct AppState { + pub db: Arc>, + pub encryption_key: [u8; 32], + pub http_client: reqwest::Client, +} + +#[cfg_attr(mobile, tauri::mobile_entry_point)] +pub fn run() { + tauri::Builder::default() + .plugin(tauri_plugin_dialog::init()) + .setup(|app| { + let db_dir = app.path().app_data_dir()?; + std::fs::create_dir_all(&db_dir)?; + + // Init database + let db_path = db_dir.join("orchai.db"); + let conn = db::init(&db_path).expect("Failed to initialize database"); + + // Load or generate encryption key + let key_path = db_dir.join("orchai.key"); + let encryption_key = load_or_generate_key(&key_path)?; + + // Shared HTTP client + let http_client = reqwest::Client::new(); + + app.manage(AppState { + db: Arc::new(Mutex::new(conn)), + encryption_key, + http_client, + }); + Ok(()) + }) + .invoke_handler(tauri::generate_handler![ + commands::project::create_project, + commands::project::list_projects, + commands::project::get_project, + commands::project::update_project, + commands::project::delete_project, + ]) + .run(tauri::generate_context!()) + .expect("error while running tauri application"); +} + +fn load_or_generate_key(path: &std::path::Path) -> Result<[u8; 32], Box> { + use rand::RngCore; + + if path.exists() { + let bytes = std::fs::read(path)?; + if bytes.len() != 32 { + return Err("Invalid key file size".into()); + } + let mut key = [0u8; 32]; + key.copy_from_slice(&bytes); + Ok(key) + } else { + let mut key = [0u8; 32]; + rand::rngs::OsRng.fill_bytes(&mut key); + std::fs::write(path, &key)?; + Ok(key) + } +} +``` + +- [x] **Step 6: Verify compilation and existing tests pass** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test +``` + +Expected: 11 tests pass (3 db + 8 project). Migration test now checks version 2. + +- [x] **Step 7: Commit** + +```bash +git add -A +git commit -m "feat: Phase 2 dependencies, migration 002, Arc AppState" +``` + +--- + +### Task 2: Crypto service + tests + +**Files:** +- Create: `src-tauri/src/services/mod.rs` +- Create: `src-tauri/src/services/crypto.rs` +- Modify: `src-tauri/src/lib.rs` (add `mod services`) + +- [x] **Step 1: Write failing tests for crypto** + +Create `src-tauri/src/services/mod.rs`: + +```rust +pub mod crypto; +``` + +Create `src-tauri/src/services/crypto.rs`: + +```rust +use aes_gcm::{ + aead::{Aead, KeyInit, OsRng}, + Aes256Gcm, Key, Nonce, +}; +use base64::{engine::general_purpose::STANDARD, Engine}; +use rand::RngCore; + +pub fn encrypt(key: &[u8; 32], plaintext: &str) -> Result { + todo!() +} + +pub fn decrypt(key: &[u8; 32], encrypted: &str) -> Result { + todo!() +} + +#[cfg(test)] +mod tests { + use super::*; + + fn test_key() -> [u8; 32] { + let mut key = [0u8; 32]; + OsRng.fill_bytes(&mut key); + key + } + + #[test] + fn test_encrypt_decrypt_roundtrip() { + let key = test_key(); + let plaintext = "my_secret_password"; + let encrypted = encrypt(&key, plaintext).expect("encrypt should succeed"); + let decrypted = decrypt(&key, &encrypted).expect("decrypt should succeed"); + assert_eq!(decrypted, plaintext); + } + + #[test] + fn test_encrypt_produces_different_ciphertext() { + let key = test_key(); + let plaintext = "same_password"; + let enc1 = encrypt(&key, plaintext).unwrap(); + let enc2 = encrypt(&key, plaintext).unwrap(); + assert_ne!(enc1, enc2, "random nonce should produce different ciphertext"); + } + + #[test] + fn test_decrypt_with_wrong_key_fails() { + let key1 = test_key(); + let key2 = test_key(); + let encrypted = encrypt(&key1, "secret").unwrap(); + let result = decrypt(&key2, &encrypted); + assert!(result.is_err()); + } + + #[test] + fn test_decrypt_invalid_base64_fails() { + let key = test_key(); + let result = decrypt(&key, "not-valid-base64!!!"); + assert!(result.is_err()); + } + + #[test] + fn test_decrypt_too_short_fails() { + let key = test_key(); + let short = STANDARD.encode(&[0u8; 5]); + let result = decrypt(&key, &short); + assert!(result.is_err()); + } + + #[test] + fn test_encrypt_empty_string() { + let key = test_key(); + let encrypted = encrypt(&key, "").expect("encrypt empty should succeed"); + let decrypted = decrypt(&key, &encrypted).expect("decrypt should succeed"); + assert_eq!(decrypted, ""); + } + + #[test] + fn test_encrypt_unicode() { + let key = test_key(); + let plaintext = "mot de passe avec accents: eaui"; + let encrypted = encrypt(&key, plaintext).unwrap(); + let decrypted = decrypt(&key, &encrypted).unwrap(); + assert_eq!(decrypted, plaintext); + } +} +``` + +Add `mod services;` to `src-tauri/src/lib.rs` (after `mod models;`). + +- [x] **Step 2: Run tests to verify they fail** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test services::crypto::tests +``` + +Expected: 7 failures with `not yet implemented`. + +- [x] **Step 3: Implement encrypt and decrypt** + +Replace the `todo!()` stubs in `src-tauri/src/services/crypto.rs`: + +```rust +pub fn encrypt(key: &[u8; 32], plaintext: &str) -> Result { + let cipher = Aes256Gcm::new(Key::::from_slice(key)); + let mut nonce_bytes = [0u8; 12]; + OsRng.fill_bytes(&mut nonce_bytes); + let nonce = Nonce::from_slice(&nonce_bytes); + + let ciphertext = cipher + .encrypt(nonce, plaintext.as_bytes()) + .map_err(|e| format!("encryption failed: {}", e))?; + + let mut combined = nonce_bytes.to_vec(); + combined.extend(ciphertext); + Ok(STANDARD.encode(&combined)) +} + +pub fn decrypt(key: &[u8; 32], encrypted: &str) -> Result { + let combined = STANDARD + .decode(encrypted) + .map_err(|e| format!("base64 decode failed: {}", e))?; + + if combined.len() < 13 { + return Err("encrypted data too short".to_string()); + } + + let (nonce_bytes, ciphertext) = combined.split_at(12); + let cipher = Aes256Gcm::new(Key::::from_slice(key)); + let nonce = Nonce::from_slice(nonce_bytes); + + let plaintext = cipher + .decrypt(nonce, ciphertext) + .map_err(|_| "decryption failed (wrong key or corrupted data)".to_string())?; + + String::from_utf8(plaintext).map_err(|e| format!("invalid UTF-8: {}", e)) +} +``` + +- [x] **Step 4: Run tests to verify they pass** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test services::crypto::tests +``` + +Expected: 7 tests pass. + +- [x] **Step 5: Commit** + +```bash +git add -A +git commit -m "feat: AES-256-GCM crypto service for credential encryption" +``` + +--- + +### Task 3: TuleapCredentials model + Tauri commands + +**Files:** +- Create: `src-tauri/src/models/credential.rs` +- Modify: `src-tauri/src/models/mod.rs` +- Create: `src-tauri/src/commands/credential.rs` +- Modify: `src-tauri/src/commands/mod.rs` +- Modify: `src-tauri/src/lib.rs` (register commands) + +- [x] **Step 1: Write failing tests for TuleapCredentials model** + +Create `src-tauri/src/models/credential.rs`: + +```rust +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct TuleapCredentials { + pub id: String, + pub tuleap_url: String, + pub username: String, + pub password_encrypted: String, +} + +/// Credential returned to frontend (no encrypted password) +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct TuleapCredentialsSafe { + pub id: String, + pub tuleap_url: String, + pub username: String, +} + +impl TuleapCredentials { + pub fn upsert( + conn: &Connection, + tuleap_url: &str, + username: &str, + password_encrypted: &str, + ) -> Result { + todo!() + } + + pub fn get(conn: &Connection) -> Result> { + todo!() + } + + pub fn delete(conn: &Connection) -> Result<()> { + todo!() + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + + fn setup() -> Connection { + db::init_in_memory().expect("db init should succeed") + } + + #[test] + fn test_upsert_creates_credentials() { + let conn = setup(); + let cred = TuleapCredentials::upsert(&conn, "https://tuleap.example.com", "user1", "encrypted_pw") + .expect("upsert should succeed"); + + assert_eq!(cred.tuleap_url, "https://tuleap.example.com"); + assert_eq!(cred.username, "user1"); + assert_eq!(cred.password_encrypted, "encrypted_pw"); + } + + #[test] + fn test_upsert_replaces_existing() { + let conn = setup(); + TuleapCredentials::upsert(&conn, "https://old.com", "old_user", "old_pw").unwrap(); + let cred = TuleapCredentials::upsert(&conn, "https://new.com", "new_user", "new_pw").unwrap(); + + assert_eq!(cred.tuleap_url, "https://new.com"); + let fetched = TuleapCredentials::get(&conn).unwrap().unwrap(); + assert_eq!(fetched.tuleap_url, "https://new.com"); + } + + #[test] + fn test_get_returns_none_when_empty() { + let conn = setup(); + let result = TuleapCredentials::get(&conn).unwrap(); + assert!(result.is_none()); + } + + #[test] + fn test_get_returns_credentials() { + let conn = setup(); + TuleapCredentials::upsert(&conn, "https://tuleap.example.com", "user1", "enc_pw").unwrap(); + let cred = TuleapCredentials::get(&conn).unwrap().unwrap(); + assert_eq!(cred.username, "user1"); + } + + #[test] + fn test_delete_removes_credentials() { + let conn = setup(); + TuleapCredentials::upsert(&conn, "https://tuleap.example.com", "user1", "enc_pw").unwrap(); + TuleapCredentials::delete(&conn).unwrap(); + let result = TuleapCredentials::get(&conn).unwrap(); + assert!(result.is_none()); + } +} +``` + +Add `pub mod credential;` to `src-tauri/src/models/mod.rs`. + +- [x] **Step 2: Run tests to verify they fail** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test models::credential::tests +``` + +Expected: 5 failures. + +- [x] **Step 3: Implement TuleapCredentials CRUD** + +Replace stubs in `src-tauri/src/models/credential.rs`: + +```rust + /// We only store one set of credentials. Upsert deletes existing and inserts new. + pub fn upsert( + conn: &Connection, + tuleap_url: &str, + username: &str, + password_encrypted: &str, + ) -> Result { + conn.execute("DELETE FROM tuleap_credentials", [])?; + let id = Uuid::new_v4().to_string(); + conn.execute( + "INSERT INTO tuleap_credentials (id, tuleap_url, username, password_encrypted) VALUES (?1, ?2, ?3, ?4)", + params![id, tuleap_url, username, password_encrypted], + )?; + Ok(TuleapCredentials { + id, + tuleap_url: tuleap_url.to_string(), + username: username.to_string(), + password_encrypted: password_encrypted.to_string(), + }) + } + + pub fn get(conn: &Connection) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, tuleap_url, username, password_encrypted FROM tuleap_credentials LIMIT 1", + )?; + let mut rows = stmt.query_map([], |row| { + Ok(TuleapCredentials { + id: row.get(0)?, + tuleap_url: row.get(1)?, + username: row.get(2)?, + password_encrypted: row.get(3)?, + }) + })?; + match rows.next() { + Some(row) => Ok(Some(row?)), + None => Ok(None), + } + } + + pub fn delete(conn: &Connection) -> Result<()> { + conn.execute("DELETE FROM tuleap_credentials", [])?; + Ok(()) + } +``` + +- [x] **Step 4: Run tests to verify they pass** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test models::credential::tests +``` + +Expected: 5 pass. + +- [x] **Step 5: Create credential Tauri commands** + +Create `src-tauri/src/commands/credential.rs`: + +```rust +use crate::error::AppError; +use crate::models::credential::{TuleapCredentials, TuleapCredentialsSafe}; +use crate::services::crypto; +use crate::AppState; +use tauri::State; + +#[tauri::command] +pub fn set_tuleap_credentials( + state: State<'_, AppState>, + tuleap_url: String, + username: String, + password: String, +) -> Result { + let encrypted = crypto::encrypt(&state.encryption_key, &password) + .map_err(|e| AppError::from(e))?; + + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let cred = TuleapCredentials::upsert(&db, &tuleap_url, &username, &encrypted)?; + + Ok(TuleapCredentialsSafe { + id: cred.id, + tuleap_url: cred.tuleap_url, + username: cred.username, + }) +} + +#[tauri::command] +pub fn get_tuleap_credentials( + state: State<'_, AppState>, +) -> Result, AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let cred = TuleapCredentials::get(&db)?; + Ok(cred.map(|c| TuleapCredentialsSafe { + id: c.id, + tuleap_url: c.tuleap_url, + username: c.username, + })) +} + +#[tauri::command] +pub fn delete_tuleap_credentials(state: State<'_, AppState>) -> Result<(), AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + TuleapCredentials::delete(&db)?; + Ok(()) +} + +#[tauri::command] +pub async fn test_tuleap_connection(state: State<'_, AppState>) -> Result { + let (base_url, username, password) = { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let cred = TuleapCredentials::get(&db)? + .ok_or_else(|| AppError::from("No credentials configured".to_string()))?; + let password = crypto::decrypt(&state.encryption_key, &cred.password_encrypted) + .map_err(|e| AppError::from(e))?; + (cred.tuleap_url, cred.username, password) + }; + + let url = format!("{}/api/projects?limit=1", base_url.trim_end_matches('/')); + let response = state.http_client + .get(&url) + .basic_auth(&username, Some(&password)) + .send() + .await?; + + if response.status().is_success() { + Ok("Connection successful".to_string()) + } else { + Err(AppError::from(format!( + "Tuleap API returned status {}", + response.status() + ))) + } +} +``` + +Add `pub mod credential;` to `src-tauri/src/commands/mod.rs`. + +- [x] **Step 6: Register commands in lib.rs** + +Add to the `invoke_handler` in `src-tauri/src/lib.rs`: + +```rust + .invoke_handler(tauri::generate_handler![ + commands::project::create_project, + commands::project::list_projects, + commands::project::get_project, + commands::project::update_project, + commands::project::delete_project, + commands::credential::set_tuleap_credentials, + commands::credential::get_tuleap_credentials, + commands::credential::delete_tuleap_credentials, + commands::credential::test_tuleap_connection, + ]) +``` + +- [x] **Step 7: Verify compilation and all tests pass** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test +``` + +Expected: all tests pass (11 existing + 5 credential + 7 crypto = 23 tests). + +- [x] **Step 8: Commit** + +```bash +git add -A +git commit -m "feat: TuleapCredentials model + encrypted storage + Tauri commands" +``` + +--- + +### Task 4: Tuleap HTTP client service + tests + +**Files:** +- Create: `src-tauri/src/services/tuleap_client.rs` +- Modify: `src-tauri/src/services/mod.rs` + +- [x] **Step 1: Write the Tuleap client with tests for parsing logic** + +Add `pub mod tuleap_client;` to `src-tauri/src/services/mod.rs`. + +Create `src-tauri/src/services/tuleap_client.rs`: + +```rust +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct TrackerField { + pub field_id: i64, + pub label: String, + pub field_type: String, + pub values: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct FieldValue { + pub id: i64, + pub label: String, +} + +pub struct TuleapClient { + http: reqwest::Client, + base_url: String, + username: String, + password: String, +} + +impl TuleapClient { + pub fn new(http: &reqwest::Client, base_url: &str, username: &str, password: &str) -> Self { + TuleapClient { + http: http.clone(), + base_url: base_url.trim_end_matches('/').to_string(), + username: username.to_string(), + password: password.to_string(), + } + } + + pub async fn test_connection(&self) -> Result<(), String> { + let url = format!("{}/api/projects?limit=1", self.base_url); + let response = self.http + .get(&url) + .basic_auth(&self.username, Some(&self.password)) + .send() + .await + .map_err(|e| format!("HTTP request failed: {}", e))?; + + if response.status().is_success() { + Ok(()) + } else { + Err(format!("Tuleap API returned status {}", response.status())) + } + } + + pub async fn get_tracker_fields(&self, tracker_id: i32) -> Result, String> { + let url = format!("{}/api/trackers/{}", self.base_url, tracker_id); + let response = self.http + .get(&url) + .basic_auth(&self.username, Some(&self.password)) + .send() + .await + .map_err(|e| format!("HTTP request failed: {}", e))?; + + if !response.status().is_success() { + return Err(format!("Tuleap API returned status {}", response.status())); + } + + let body: serde_json::Value = response.json().await + .map_err(|e| format!("Failed to parse response: {}", e))?; + + Ok(parse_tracker_fields(&body)) + } + + pub async fn get_artifacts(&self, tracker_id: i32) -> Result, String> { + let mut all_artifacts = Vec::new(); + let mut offset = 0; + let limit = 100; + + loop { + let url = format!( + "{}/api/trackers/{}/artifacts?limit={}&offset={}&values=all", + self.base_url, tracker_id, limit, offset + ); + let response = self.http + .get(&url) + .basic_auth(&self.username, Some(&self.password)) + .send() + .await + .map_err(|e| format!("HTTP request failed: {}", e))?; + + if !response.status().is_success() { + return Err(format!("Tuleap API returned status {}", response.status())); + } + + let total: i64 = response.headers() + .get("x-pagination-size") + .and_then(|v| v.to_str().ok()) + .and_then(|v| v.parse().ok()) + .unwrap_or(0); + + let artifacts: Vec = response.json().await + .map_err(|e| format!("Failed to parse response: {}", e))?; + + let count = artifacts.len(); + all_artifacts.extend(artifacts); + + offset += count as i64; + if offset >= total || count == 0 { + break; + } + } + + Ok(all_artifacts) + } +} + +/// Parse tracker fields from the Tuleap API tracker response +pub fn parse_tracker_fields(tracker_json: &serde_json::Value) -> Vec { + let fields = match tracker_json.get("fields") { + Some(serde_json::Value::Array(arr)) => arr, + _ => return Vec::new(), + }; + + fields.iter().filter_map(|field| { + let field_id = field.get("field_id")?.as_i64()?; + let label = field.get("label")?.as_str()?.to_string(); + let field_type = field.get("type")?.as_str()?.to_string(); + + // Only include fields with selectable values + let values = match field_type.as_str() { + "sb" | "msb" | "rb" | "cb" => { + extract_field_values(field) + } + _ => Vec::new(), + }; + + Some(TrackerField { + field_id, + label, + field_type, + values, + }) + }).collect() +} + +fn extract_field_values(field: &serde_json::Value) -> Vec { + // Try "values" array first (used by sb, rb) + if let Some(serde_json::Value::Array(vals)) = field.get("values") { + return vals.iter().filter_map(|v| { + let id = v.get("id")?.as_i64()?; + let label = v.get("label")?.as_str()?.to_string(); + if label == "None" { return None; } + Some(FieldValue { id, label }) + }).collect(); + } + // Try "bind_value_objects" (used by msb) + if let Some(serde_json::Value::Array(vals)) = field.get("bind_value_objects") { + return vals.iter().filter_map(|v| { + let id = v.get("id")?.as_i64()?; + let label = v.get("display_name") + .or_else(|| v.get("label")) + .and_then(|l| l.as_str()) + .map(String::from)?; + Some(FieldValue { id, label }) + }).collect(); + } + Vec::new() +} + +/// Extract the displayable values of a field from an artifact's "values" array. +/// Returns the list of labels/names for the field. Used by the filter engine. +pub fn extract_artifact_field_values(artifact: &serde_json::Value, field_label: &str) -> Vec { + let values = match artifact.get("values") { + Some(serde_json::Value::Array(arr)) => arr, + _ => return Vec::new(), + }; + + for field in values { + let label = match field.get("label").and_then(|l| l.as_str()) { + Some(l) => l, + None => continue, + }; + if label != field_label { + continue; + } + + let field_type = field.get("type").and_then(|t| t.as_str()).unwrap_or(""); + + return match field_type { + "sb" | "rb" => { + // Select box / radio button: values[*].label + field.get("values") + .and_then(|v| v.as_array()) + .map(|arr| arr.iter() + .filter_map(|v| v.get("label").and_then(|l| l.as_str()).map(String::from)) + .collect()) + .unwrap_or_default() + } + "msb" | "cb" => { + // Multi-select / checkbox: bind_value_objects[*].display_name or label + field.get("bind_value_objects") + .and_then(|v| v.as_array()) + .map(|arr| arr.iter() + .filter_map(|v| { + v.get("display_name") + .or_else(|| v.get("label")) + .and_then(|l| l.as_str()) + .map(String::from) + }) + .collect()) + .unwrap_or_default() + } + "string" | "text" | "int" | "float" => { + field.get("value") + .and_then(|v| match v { + serde_json::Value::String(s) => Some(s.clone()), + serde_json::Value::Number(n) => Some(n.to_string()), + _ => None, + }) + .into_iter() + .collect() + } + _ => Vec::new(), + }; + } + Vec::new() +} + +#[cfg(test)] +mod tests { + use super::*; + use serde_json::json; + + #[test] + fn test_parse_tracker_fields_extracts_sb() { + let tracker = json!({ + "fields": [ + { + "field_id": 1, + "label": "Status", + "type": "sb", + "values": [ + { "id": 100, "label": "None" }, + { "id": 101, "label": "Nouveau" }, + { "id": 102, "label": "En cours" } + ] + } + ] + }); + let fields = parse_tracker_fields(&tracker); + assert_eq!(fields.len(), 1); + assert_eq!(fields[0].label, "Status"); + assert_eq!(fields[0].values.len(), 2); // "None" filtered out + assert_eq!(fields[0].values[0].label, "Nouveau"); + } + + #[test] + fn test_parse_tracker_fields_extracts_msb() { + let tracker = json!({ + "fields": [ + { + "field_id": 5, + "label": "Assigned to", + "type": "msb", + "bind_value_objects": [ + { "id": 200, "display_name": "Alice" }, + { "id": 201, "display_name": "Bob" } + ] + } + ] + }); + let fields = parse_tracker_fields(&tracker); + assert_eq!(fields.len(), 1); + assert_eq!(fields[0].values.len(), 2); + assert_eq!(fields[0].values[0].label, "Alice"); + } + + #[test] + fn test_parse_tracker_fields_skips_text_fields() { + let tracker = json!({ + "fields": [ + { "field_id": 10, "label": "Description", "type": "text" } + ] + }); + let fields = parse_tracker_fields(&tracker); + assert_eq!(fields.len(), 1); + assert!(fields[0].values.is_empty()); // text fields have no selectable values + } + + #[test] + fn test_extract_artifact_field_values_sb() { + let artifact = json!({ + "values": [ + { + "field_id": 1, + "label": "Status", + "type": "sb", + "values": [{ "id": 101, "label": "Nouveau" }] + } + ] + }); + let vals = extract_artifact_field_values(&artifact, "Status"); + assert_eq!(vals, vec!["Nouveau"]); + } + + #[test] + fn test_extract_artifact_field_values_msb() { + let artifact = json!({ + "values": [ + { + "field_id": 5, + "label": "Assigned to", + "type": "msb", + "bind_value_objects": [ + { "id": 200, "display_name": "Team Maintenance" } + ] + } + ] + }); + let vals = extract_artifact_field_values(&artifact, "Assigned to"); + assert_eq!(vals, vec!["Team Maintenance"]); + } + + #[test] + fn test_extract_artifact_field_values_missing_field() { + let artifact = json!({ "values": [] }); + let vals = extract_artifact_field_values(&artifact, "Nonexistent"); + assert!(vals.is_empty()); + } + + #[test] + fn test_extract_artifact_field_values_string_field() { + let artifact = json!({ + "values": [ + { "field_id": 20, "label": "Summary", "type": "string", "value": "Login broken" } + ] + }); + let vals = extract_artifact_field_values(&artifact, "Summary"); + assert_eq!(vals, vec!["Login broken"]); + } +} +``` + +- [x] **Step 2: Run tests** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test services::tuleap_client::tests +``` + +Expected: 7 tests pass. + +- [x] **Step 3: Commit** + +```bash +git add -A +git commit -m "feat: Tuleap HTTP client with artifact parsing and field extraction" +``` + +--- + +### Task 5: WatchedTracker model + CRUD + tests + +**Files:** +- Create: `src-tauri/src/models/tracker.rs` +- Modify: `src-tauri/src/models/mod.rs` + +- [x] **Step 1: Write failing tests for WatchedTracker** + +Add `pub mod tracker;` to `src-tauri/src/models/mod.rs`. + +Create `src-tauri/src/models/tracker.rs`: + +```rust +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct AgentConfig { + pub analyst_command: String, + pub analyst_args: Vec, + pub developer_command: String, + pub developer_args: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct FilterGroup { + pub conditions: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Filter { + pub field: String, + pub operator: String, // "In", "NotIn", "Equals", "NotEquals" + pub value: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct WatchedTracker { + pub id: String, + pub project_id: String, + pub tracker_id: i32, + pub tracker_label: String, + pub polling_interval: i32, + pub agent_config: AgentConfig, + pub filters: Vec, + pub enabled: bool, + pub last_polled_at: Option, + pub created_at: String, +} + +impl WatchedTracker { + pub fn insert( + conn: &Connection, + project_id: &str, + tracker_id: i32, + tracker_label: &str, + polling_interval: i32, + agent_config: &AgentConfig, + filters: &[FilterGroup], + ) -> Result { + todo!() + } + + pub fn list_by_project(conn: &Connection, project_id: &str) -> Result> { + todo!() + } + + pub fn list_all_enabled(conn: &Connection) -> Result> { + todo!() + } + + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + todo!() + } + + pub fn update( + conn: &Connection, + id: &str, + polling_interval: i32, + agent_config: &AgentConfig, + filters: &[FilterGroup], + enabled: bool, + ) -> Result<()> { + todo!() + } + + pub fn update_last_polled(conn: &Connection, id: &str) -> Result<()> { + todo!() + } + + pub fn delete(conn: &Connection, id: &str) -> Result<()> { + todo!() + } +} + +fn from_row(row: &rusqlite::Row) -> rusqlite::Result { + let agent_config_json: String = row.get(5)?; + let filters_json: String = row.get(6)?; + let enabled: i32 = row.get(7)?; + + Ok(WatchedTracker { + id: row.get(0)?, + project_id: row.get(1)?, + tracker_id: row.get(2)?, + tracker_label: row.get(3)?, + polling_interval: row.get(4)?, + agent_config: serde_json::from_str(&agent_config_json).unwrap_or(AgentConfig { + analyst_command: String::new(), + analyst_args: Vec::new(), + developer_command: String::new(), + developer_args: Vec::new(), + }), + filters: serde_json::from_str(&filters_json).unwrap_or_default(), + enabled: enabled == 1, + last_polled_at: row.get(8)?, + created_at: row.get(9)?, + }) +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + use crate::models::project::Project; + + fn setup() -> Connection { + let conn = db::init_in_memory().expect("db init should succeed"); + Project::insert(&conn, "Test Project", "/path/test", None, "main").unwrap(); + conn + } + + fn get_project_id(conn: &Connection) -> String { + Project::list(conn).unwrap()[0].id.clone() + } + + fn default_agent_config() -> AgentConfig { + AgentConfig { + analyst_command: "claude".to_string(), + analyst_args: vec!["--print".to_string()], + developer_command: "claude".to_string(), + developer_args: vec!["--print".to_string()], + } + } + + #[test] + fn test_insert_tracker() { + let conn = setup(); + let project_id = get_project_id(&conn); + let tracker = WatchedTracker::insert( + &conn, &project_id, 456, "Bugs", 10, &default_agent_config(), &[], + ).expect("insert should succeed"); + + assert_eq!(tracker.tracker_id, 456); + assert_eq!(tracker.tracker_label, "Bugs"); + assert_eq!(tracker.polling_interval, 10); + assert!(tracker.enabled); + } + + #[test] + fn test_list_by_project() { + let conn = setup(); + let project_id = get_project_id(&conn); + WatchedTracker::insert(&conn, &project_id, 1, "Bugs", 10, &default_agent_config(), &[]).unwrap(); + WatchedTracker::insert(&conn, &project_id, 2, "Tasks", 15, &default_agent_config(), &[]).unwrap(); + + let trackers = WatchedTracker::list_by_project(&conn, &project_id).unwrap(); + assert_eq!(trackers.len(), 2); + } + + #[test] + fn test_list_all_enabled() { + let conn = setup(); + let project_id = get_project_id(&conn); + let t = WatchedTracker::insert(&conn, &project_id, 1, "Bugs", 10, &default_agent_config(), &[]).unwrap(); + WatchedTracker::insert(&conn, &project_id, 2, "Tasks", 15, &default_agent_config(), &[]).unwrap(); + WatchedTracker::update(&conn, &t.id, 10, &default_agent_config(), &[], false).unwrap(); + + let enabled = WatchedTracker::list_all_enabled(&conn).unwrap(); + assert_eq!(enabled.len(), 1); + assert_eq!(enabled[0].tracker_label, "Tasks"); + } + + #[test] + fn test_get_by_id() { + let conn = setup(); + let project_id = get_project_id(&conn); + let created = WatchedTracker::insert(&conn, &project_id, 456, "Bugs", 10, &default_agent_config(), &[]).unwrap(); + let found = WatchedTracker::get_by_id(&conn, &created.id).unwrap(); + assert_eq!(found.tracker_id, 456); + } + + #[test] + fn test_update_tracker() { + let conn = setup(); + let project_id = get_project_id(&conn); + let created = WatchedTracker::insert(&conn, &project_id, 456, "Bugs", 10, &default_agent_config(), &[]).unwrap(); + + let filters = vec![FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "In".to_string(), + value: vec!["Nouveau".to_string()], + }], + }]; + WatchedTracker::update(&conn, &created.id, 20, &default_agent_config(), &filters, true).unwrap(); + + let updated = WatchedTracker::get_by_id(&conn, &created.id).unwrap(); + assert_eq!(updated.polling_interval, 20); + assert_eq!(updated.filters.len(), 1); + assert_eq!(updated.filters[0].conditions[0].field, "Status"); + } + + #[test] + fn test_update_last_polled() { + let conn = setup(); + let project_id = get_project_id(&conn); + let created = WatchedTracker::insert(&conn, &project_id, 456, "Bugs", 10, &default_agent_config(), &[]).unwrap(); + assert!(created.last_polled_at.is_none()); + + WatchedTracker::update_last_polled(&conn, &created.id).unwrap(); + let updated = WatchedTracker::get_by_id(&conn, &created.id).unwrap(); + assert!(updated.last_polled_at.is_some()); + } + + #[test] + fn test_delete_tracker() { + let conn = setup(); + let project_id = get_project_id(&conn); + let created = WatchedTracker::insert(&conn, &project_id, 456, "Bugs", 10, &default_agent_config(), &[]).unwrap(); + WatchedTracker::delete(&conn, &created.id).unwrap(); + assert!(WatchedTracker::get_by_id(&conn, &created.id).is_err()); + } +} +``` + +- [x] **Step 2: Run tests to verify they fail** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test models::tracker::tests +``` + +Expected: 7 failures. + +- [x] **Step 3: Implement WatchedTracker CRUD** + +Replace the `todo!()` stubs in `src-tauri/src/models/tracker.rs`: + +```rust + pub fn insert( + conn: &Connection, + project_id: &str, + tracker_id: i32, + tracker_label: &str, + polling_interval: i32, + agent_config: &AgentConfig, + filters: &[FilterGroup], + ) -> Result { + let id = Uuid::new_v4().to_string(); + let now = chrono::Utc::now().to_rfc3339(); + let agent_config_json = serde_json::to_string(agent_config).unwrap(); + let filters_json = serde_json::to_string(filters).unwrap(); + + conn.execute( + "INSERT INTO watched_trackers (id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, created_at) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8)", + params![id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, now], + )?; + + Ok(WatchedTracker { + id, + project_id: project_id.to_string(), + tracker_id, + tracker_label: tracker_label.to_string(), + polling_interval, + agent_config: agent_config.clone(), + filters: filters.to_vec(), + enabled: true, + last_polled_at: None, + created_at: now, + }) + } + + pub fn list_by_project(conn: &Connection, project_id: &str) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, enabled, last_polled_at, created_at FROM watched_trackers WHERE project_id = ?1 ORDER BY created_at DESC", + )?; + let rows = stmt.query_map(params![project_id], |row| from_row(row))?; + rows.collect() + } + + pub fn list_all_enabled(conn: &Connection) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, enabled, last_polled_at, created_at FROM watched_trackers WHERE enabled = 1", + )?; + let rows = stmt.query_map([], |row| from_row(row))?; + rows.collect() + } + + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + conn.query_row( + "SELECT id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, enabled, last_polled_at, created_at FROM watched_trackers WHERE id = ?1", + params![id], + |row| from_row(row), + ) + } + + pub fn update( + conn: &Connection, + id: &str, + polling_interval: i32, + agent_config: &AgentConfig, + filters: &[FilterGroup], + enabled: bool, + ) -> Result<()> { + let agent_config_json = serde_json::to_string(agent_config).unwrap(); + let filters_json = serde_json::to_string(filters).unwrap(); + let enabled_int: i32 = if enabled { 1 } else { 0 }; + + let affected = conn.execute( + "UPDATE watched_trackers SET polling_interval = ?1, agent_config_json = ?2, filters_json = ?3, enabled = ?4 WHERE id = ?5", + params![polling_interval, agent_config_json, filters_json, enabled_int, id], + )?; + if affected == 0 { + return Err(rusqlite::Error::QueryReturnedNoRows); + } + Ok(()) + } + + pub fn update_last_polled(conn: &Connection, id: &str) -> Result<()> { + let now = chrono::Utc::now().to_rfc3339(); + conn.execute( + "UPDATE watched_trackers SET last_polled_at = ?1 WHERE id = ?2", + params![now, id], + )?; + Ok(()) + } + + pub fn delete(conn: &Connection, id: &str) -> Result<()> { + let affected = conn.execute("DELETE FROM watched_trackers WHERE id = ?1", params![id])?; + if affected == 0 { + return Err(rusqlite::Error::QueryReturnedNoRows); + } + Ok(()) + } +``` + +- [x] **Step 4: Run tests to verify they pass** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test models::tracker::tests +``` + +Expected: 7 pass. + +- [x] **Step 5: Commit** + +```bash +git add -A +git commit -m "feat: WatchedTracker model with CRUD, filters, and agent config" +``` + +--- + +### Task 6: Filter engine + tests + +**Files:** +- Create: `src-tauri/src/services/filter_engine.rs` +- Modify: `src-tauri/src/services/mod.rs` + +- [x] **Step 1: Write the filter engine with tests** + +Add `pub mod filter_engine;` to `src-tauri/src/services/mod.rs`. + +Create `src-tauri/src/services/filter_engine.rs`: + +```rust +use crate::models::tracker::{Filter, FilterGroup}; +use crate::services::tuleap_client::extract_artifact_field_values; + +/// Filter artifacts using AND/OR logic. +/// FilterGroups are combined with AND. +/// Conditions within a FilterGroup are combined with OR. +/// Returns only artifacts matching ALL groups. +pub fn apply_filters( + artifacts: &[serde_json::Value], + filter_groups: &[FilterGroup], +) -> Vec { + if filter_groups.is_empty() { + return artifacts.to_vec(); + } + + artifacts.iter() + .filter(|artifact| matches_all_groups(artifact, filter_groups)) + .cloned() + .collect() +} + +fn matches_all_groups(artifact: &serde_json::Value, groups: &[FilterGroup]) -> bool { + groups.iter().all(|group| matches_any_condition(artifact, &group.conditions)) +} + +fn matches_any_condition(artifact: &serde_json::Value, conditions: &[Filter]) -> bool { + if conditions.is_empty() { + return true; + } + conditions.iter().any(|condition| matches_condition(artifact, condition)) +} + +fn matches_condition(artifact: &serde_json::Value, condition: &Filter) -> bool { + let field_values = extract_artifact_field_values(artifact, &condition.field); + + match condition.operator.as_str() { + "Equals" => { + condition.value.len() == 1 + && field_values.iter().any(|v| v == &condition.value[0]) + } + "NotEquals" => { + condition.value.len() == 1 + && !field_values.iter().any(|v| v == &condition.value[0]) + } + "In" => { + field_values.iter().any(|v| condition.value.contains(v)) + } + "NotIn" => { + !field_values.iter().any(|v| condition.value.contains(v)) + } + _ => false, + } +} + +#[cfg(test)] +mod tests { + use super::*; + use serde_json::json; + + fn make_artifact(status: &str, assigned: &str, priority: &str) -> serde_json::Value { + json!({ + "id": 123, + "title": "Test ticket", + "values": [ + { + "field_id": 1, "label": "Status", "type": "sb", + "values": [{ "id": 1, "label": status }] + }, + { + "field_id": 2, "label": "Assigned to", "type": "msb", + "bind_value_objects": [{ "id": 2, "display_name": assigned }] + }, + { + "field_id": 3, "label": "Priority", "type": "sb", + "values": [{ "id": 3, "label": priority }] + } + ] + }) + } + + #[test] + fn test_empty_filters_returns_all() { + let artifacts = vec![make_artifact("Nouveau", "Alice", "Haute")]; + let result = apply_filters(&artifacts, &[]); + assert_eq!(result.len(), 1); + } + + #[test] + fn test_single_in_filter() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Ferme", "Bob", "Basse"), + ]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "In".to_string(), + value: vec!["Nouveau".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 1); + } + + #[test] + fn test_or_within_group() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("A traiter", "Bob", "Basse"), + make_artifact("Ferme", "Charlie", "Moyenne"), + ]; + let groups = vec![FilterGroup { + conditions: vec![ + Filter { field: "Status".to_string(), operator: "In".to_string(), value: vec!["Nouveau".to_string()] }, + Filter { field: "Status".to_string(), operator: "In".to_string(), value: vec!["A traiter".to_string()] }, + ], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 2); + } + + #[test] + fn test_and_across_groups() { + // (Status Nouveau OR A traiter) AND (Assigned to Team Maintenance) + let artifacts = vec![ + make_artifact("Nouveau", "Team Maintenance", "Haute"), + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Ferme", "Team Maintenance", "Basse"), + ]; + let groups = vec![ + FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "In".to_string(), + value: vec!["Nouveau".to_string(), "A traiter".to_string()], + }], + }, + FilterGroup { + conditions: vec![Filter { + field: "Assigned to".to_string(), + operator: "In".to_string(), + value: vec!["Team Maintenance".to_string()], + }], + }, + ]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 1); + assert_eq!(result[0]["values"][0]["values"][0]["label"], "Nouveau"); + } + + #[test] + fn test_not_in_filter() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Ferme", "Bob", "Basse"), + ]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "NotIn".to_string(), + value: vec!["Ferme".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 1); + } + + #[test] + fn test_equals_filter() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Nouveau", "Bob", "Basse"), + ]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Priority".to_string(), + operator: "Equals".to_string(), + value: vec!["Haute".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 1); + } + + #[test] + fn test_no_match_returns_empty() { + let artifacts = vec![make_artifact("Ferme", "Alice", "Basse")]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "Equals".to_string(), + value: vec!["Nouveau".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert!(result.is_empty()); + } +} +``` + +- [x] **Step 2: Run tests** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test services::filter_engine::tests +``` + +Expected: 7 pass. + +- [x] **Step 3: Commit** + +```bash +git add -A +git commit -m "feat: AND/OR filter engine for Tuleap artifact filtering" +``` + +--- + +### Task 7: ProcessedTicket model + tests + +**Files:** +- Create: `src-tauri/src/models/ticket.rs` +- Modify: `src-tauri/src/models/mod.rs` + +- [x] **Step 1: Write failing tests** + +Add `pub mod ticket;` to `src-tauri/src/models/mod.rs`. + +Create `src-tauri/src/models/ticket.rs`: + +```rust +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct ProcessedTicket { + pub id: String, + pub tracker_id: String, + pub artifact_id: i32, + pub artifact_title: String, + pub artifact_data: String, + pub status: String, + pub analyst_report: Option, + pub developer_report: Option, + pub worktree_path: Option, + pub branch_name: Option, + pub detected_at: String, + pub processed_at: Option, +} + +impl ProcessedTicket { + /// Insert a new ticket if it hasn't been processed before. + /// Returns Some(ticket) if inserted, None if already exists. + pub fn insert_if_new( + conn: &Connection, + tracker_id: &str, + artifact_id: i32, + artifact_title: &str, + artifact_data: &str, + ) -> Result> { + todo!() + } + + /// Check if an artifact has already been processed for this tracker. + pub fn exists(conn: &Connection, tracker_id: &str, artifact_id: i32) -> Result { + todo!() + } + + pub fn list_by_tracker(conn: &Connection, tracker_id: &str) -> Result> { + todo!() + } + + pub fn list_by_project(conn: &Connection, project_id: &str) -> Result> { + todo!() + } + + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + todo!() + } +} + +fn from_row(row: &rusqlite::Row) -> rusqlite::Result { + Ok(ProcessedTicket { + id: row.get(0)?, + tracker_id: row.get(1)?, + artifact_id: row.get(2)?, + artifact_title: row.get(3)?, + artifact_data: row.get(4)?, + status: row.get(5)?, + analyst_report: row.get(6)?, + developer_report: row.get(7)?, + worktree_path: row.get(8)?, + branch_name: row.get(9)?, + detected_at: row.get(10)?, + processed_at: row.get(11)?, + }) +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + use crate::models::project::Project; + use crate::models::tracker::{AgentConfig, WatchedTracker}; + + fn setup() -> (Connection, String) { + let conn = db::init_in_memory().expect("db init should succeed"); + let project = Project::insert(&conn, "Test", "/path", None, "main").unwrap(); + let agent_config = AgentConfig { + analyst_command: "claude".to_string(), + analyst_args: vec![], + developer_command: "claude".to_string(), + developer_args: vec![], + }; + let tracker = WatchedTracker::insert(&conn, &project.id, 456, "Bugs", 10, &agent_config, &[]).unwrap(); + (conn, tracker.id) + } + + #[test] + fn test_insert_if_new_creates_ticket() { + let (conn, tracker_id) = setup(); + let ticket = ProcessedTicket::insert_if_new(&conn, &tracker_id, 123, "Bug title", "{}") + .expect("should succeed"); + assert!(ticket.is_some()); + let t = ticket.unwrap(); + assert_eq!(t.artifact_id, 123); + assert_eq!(t.status, "Pending"); + } + + #[test] + fn test_insert_if_new_returns_none_for_duplicate() { + let (conn, tracker_id) = setup(); + ProcessedTicket::insert_if_new(&conn, &tracker_id, 123, "Bug", "{}").unwrap(); + let result = ProcessedTicket::insert_if_new(&conn, &tracker_id, 123, "Bug", "{}").unwrap(); + assert!(result.is_none()); + } + + #[test] + fn test_exists() { + let (conn, tracker_id) = setup(); + assert!(!ProcessedTicket::exists(&conn, &tracker_id, 123).unwrap()); + ProcessedTicket::insert_if_new(&conn, &tracker_id, 123, "Bug", "{}").unwrap(); + assert!(ProcessedTicket::exists(&conn, &tracker_id, 123).unwrap()); + } + + #[test] + fn test_list_by_tracker() { + let (conn, tracker_id) = setup(); + ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "Bug 1", "{}").unwrap(); + ProcessedTicket::insert_if_new(&conn, &tracker_id, 2, "Bug 2", "{}").unwrap(); + let tickets = ProcessedTicket::list_by_tracker(&conn, &tracker_id).unwrap(); + assert_eq!(tickets.len(), 2); + } + + #[test] + fn test_get_by_id() { + let (conn, tracker_id) = setup(); + let created = ProcessedTicket::insert_if_new(&conn, &tracker_id, 123, "Bug", "{}").unwrap().unwrap(); + let found = ProcessedTicket::get_by_id(&conn, &created.id).unwrap(); + assert_eq!(found.artifact_id, 123); + } +} +``` + +- [x] **Step 2: Run tests to verify they fail** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test models::ticket::tests +``` + +Expected: 5 failures. + +- [x] **Step 3: Implement ProcessedTicket methods** + +Replace stubs in `src-tauri/src/models/ticket.rs`: + +```rust + pub fn insert_if_new( + conn: &Connection, + tracker_id: &str, + artifact_id: i32, + artifact_title: &str, + artifact_data: &str, + ) -> Result> { + if Self::exists(conn, tracker_id, artifact_id)? { + return Ok(None); + } + let id = Uuid::new_v4().to_string(); + let now = chrono::Utc::now().to_rfc3339(); + conn.execute( + "INSERT INTO processed_tickets (id, tracker_id, artifact_id, artifact_title, artifact_data, status, detected_at) VALUES (?1, ?2, ?3, ?4, ?5, 'Pending', ?6)", + params![id, tracker_id, artifact_id, artifact_title, artifact_data, now], + )?; + Ok(Some(ProcessedTicket { + id, + tracker_id: tracker_id.to_string(), + artifact_id, + artifact_title: artifact_title.to_string(), + artifact_data: artifact_data.to_string(), + status: "Pending".to_string(), + analyst_report: None, + developer_report: None, + worktree_path: None, + branch_name: None, + detected_at: now, + processed_at: None, + })) + } + + pub fn exists(conn: &Connection, tracker_id: &str, artifact_id: i32) -> Result { + let count: i32 = conn.query_row( + "SELECT COUNT(*) FROM processed_tickets WHERE tracker_id = ?1 AND artifact_id = ?2", + params![tracker_id, artifact_id], + |row| row.get(0), + )?; + Ok(count > 0) + } + + pub fn list_by_tracker(conn: &Connection, tracker_id: &str) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, tracker_id, artifact_id, artifact_title, artifact_data, status, analyst_report, developer_report, worktree_path, branch_name, detected_at, processed_at FROM processed_tickets WHERE tracker_id = ?1 ORDER BY detected_at DESC", + )?; + let rows = stmt.query_map(params![tracker_id], |row| from_row(row))?; + rows.collect() + } + + pub fn list_by_project(conn: &Connection, project_id: &str) -> Result> { + let mut stmt = conn.prepare( + "SELECT pt.id, pt.tracker_id, pt.artifact_id, pt.artifact_title, pt.artifact_data, pt.status, pt.analyst_report, pt.developer_report, pt.worktree_path, pt.branch_name, pt.detected_at, pt.processed_at FROM processed_tickets pt INNER JOIN watched_trackers wt ON pt.tracker_id = wt.id WHERE wt.project_id = ?1 ORDER BY pt.detected_at DESC", + )?; + let rows = stmt.query_map(params![project_id], |row| from_row(row))?; + rows.collect() + } + + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + conn.query_row( + "SELECT id, tracker_id, artifact_id, artifact_title, artifact_data, status, analyst_report, developer_report, worktree_path, branch_name, detected_at, processed_at FROM processed_tickets WHERE id = ?1", + params![id], + |row| from_row(row), + ) + } +``` + +- [x] **Step 4: Run tests** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test models::ticket::tests +``` + +Expected: 5 pass. + +- [x] **Step 5: Commit** + +```bash +git add -A +git commit -m "feat: ProcessedTicket model with deduplication for new ticket detection" +``` + +--- + +### Task 8: Tracker + poller Tauri commands + +**Files:** +- Create: `src-tauri/src/commands/tracker.rs` +- Create: `src-tauri/src/commands/poller.rs` +- Modify: `src-tauri/src/commands/mod.rs` +- Modify: `src-tauri/src/lib.rs` (register commands) + +- [x] **Step 1: Create tracker commands** + +Create `src-tauri/src/commands/tracker.rs`: + +```rust +use crate::error::AppError; +use crate::models::credential::TuleapCredentials; +use crate::models::tracker::{AgentConfig, FilterGroup, WatchedTracker}; +use crate::models::ticket::ProcessedTicket; +use crate::services::crypto; +use crate::services::tuleap_client::TuleapClient; +use crate::AppState; +use tauri::State; + +#[tauri::command] +pub fn add_tracker( + state: State<'_, AppState>, + project_id: String, + tracker_id: i32, + tracker_label: String, + polling_interval: i32, + agent_config: AgentConfig, + filters: Vec, +) -> Result { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let tracker = WatchedTracker::insert(&db, &project_id, tracker_id, &tracker_label, polling_interval, &agent_config, &filters)?; + Ok(tracker) +} + +#[tauri::command] +pub fn list_trackers( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let trackers = WatchedTracker::list_by_project(&db, &project_id)?; + Ok(trackers) +} + +#[tauri::command] +pub fn update_tracker( + state: State<'_, AppState>, + id: String, + polling_interval: i32, + agent_config: AgentConfig, + filters: Vec, + enabled: bool, +) -> Result<(), AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + WatchedTracker::update(&db, &id, polling_interval, &agent_config, &filters, enabled)?; + Ok(()) +} + +#[tauri::command] +pub fn remove_tracker(state: State<'_, AppState>, id: String) -> Result<(), AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + WatchedTracker::delete(&db, &id)?; + Ok(()) +} + +#[tauri::command] +pub async fn get_tracker_fields( + state: State<'_, AppState>, + tracker_id: i32, +) -> Result, AppError> { + let client = build_tuleap_client(&state)?; + let fields = client.get_tracker_fields(tracker_id).await + .map_err(|e| AppError::from(e))?; + Ok(fields) +} + +#[tauri::command] +pub fn list_processed_tickets( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let tickets = ProcessedTicket::list_by_project(&db, &project_id)?; + Ok(tickets) +} + +fn build_tuleap_client(state: &State<'_, AppState>) -> Result { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let cred = TuleapCredentials::get(&db)? + .ok_or_else(|| AppError::from("No Tuleap credentials configured".to_string()))?; + let password = crypto::decrypt(&state.encryption_key, &cred.password_encrypted) + .map_err(|e| AppError::from(e))?; + Ok(TuleapClient::new(&state.http_client, &cred.tuleap_url, &cred.username, &password)) +} +``` + +- [x] **Step 2: Create poller commands** + +Create `src-tauri/src/commands/poller.rs`: + +```rust +use crate::error::AppError; +use crate::models::credential::TuleapCredentials; +use crate::models::ticket::ProcessedTicket; +use crate::models::tracker::WatchedTracker; +use crate::services::{crypto, filter_engine}; +use crate::services::tuleap_client::TuleapClient; +use crate::AppState; +use tauri::State; + +#[tauri::command] +pub async fn manual_poll( + state: State<'_, AppState>, + tracker_id: String, +) -> Result, AppError> { + let (tracker, client) = { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let tracker = WatchedTracker::get_by_id(&db, &tracker_id)?; + let cred = TuleapCredentials::get(&db)? + .ok_or_else(|| AppError::from("No Tuleap credentials configured".to_string()))?; + let password = crypto::decrypt(&state.encryption_key, &cred.password_encrypted) + .map_err(|e| AppError::from(e))?; + let client = TuleapClient::new(&state.http_client, &cred.tuleap_url, &cred.username, &password); + (tracker, client) + }; + + let artifacts = client.get_artifacts(tracker.tracker_id).await + .map_err(|e| AppError::from(e))?; + + let filtered = filter_engine::apply_filters(&artifacts, &tracker.filters); + + let mut new_tickets = Vec::new(); + { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + for artifact in &filtered { + let artifact_id = artifact.get("id").and_then(|v| v.as_i64()).unwrap_or(0) as i32; + let artifact_title = artifact.get("title").and_then(|v| v.as_str()).unwrap_or("Untitled"); + let artifact_data = serde_json::to_string(artifact).unwrap_or_default(); + + if let Some(ticket) = ProcessedTicket::insert_if_new( + &db, &tracker_id, artifact_id, artifact_title, &artifact_data, + )? { + new_tickets.push(ticket); + } + } + WatchedTracker::update_last_polled(&db, &tracker_id)?; + } + + Ok(new_tickets) +} + +#[tauri::command] +pub fn get_queue_status( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let tickets = ProcessedTicket::list_by_project(&db, &project_id)?; + Ok(tickets) +} +``` + +- [x] **Step 3: Update commands/mod.rs** + +Replace `src-tauri/src/commands/mod.rs`: + +```rust +pub mod credential; +pub mod poller; +pub mod project; +pub mod tracker; +``` + +- [x] **Step 4: Register all new commands in lib.rs** + +Update the `invoke_handler` in `src-tauri/src/lib.rs`: + +```rust + .invoke_handler(tauri::generate_handler![ + commands::project::create_project, + commands::project::list_projects, + commands::project::get_project, + commands::project::update_project, + commands::project::delete_project, + commands::credential::set_tuleap_credentials, + commands::credential::get_tuleap_credentials, + commands::credential::delete_tuleap_credentials, + commands::credential::test_tuleap_connection, + commands::tracker::add_tracker, + commands::tracker::list_trackers, + commands::tracker::update_tracker, + commands::tracker::remove_tracker, + commands::tracker::get_tracker_fields, + commands::tracker::list_processed_tickets, + commands::poller::manual_poll, + commands::poller::get_queue_status, + ]) +``` + +- [x] **Step 5: Verify compilation and all tests pass** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test +``` + +Expected: all tests pass. + +- [x] **Step 6: Commit** + +```bash +git add -A +git commit -m "feat: Tauri commands for tracker CRUD, Tuleap fields, and manual polling" +``` + +--- + +### Task 9: Background poller service + +**Files:** +- Create: `src-tauri/src/services/poller.rs` +- Modify: `src-tauri/src/services/mod.rs` +- Modify: `src-tauri/src/lib.rs` (spawn poller on startup) + +- [x] **Step 1: Create poller service** + +Add `pub mod poller;` to `src-tauri/src/services/mod.rs`. + +Create `src-tauri/src/services/poller.rs`: + +```rust +use crate::models::credential::TuleapCredentials; +use crate::models::ticket::ProcessedTicket; +use crate::models::tracker::WatchedTracker; +use crate::services::{crypto, filter_engine}; +use crate::services::tuleap_client::TuleapClient; +use rusqlite::Connection; +use std::sync::{Arc, Mutex}; +use tauri::{AppHandle, Emitter}; +use tokio::time::{interval, Duration}; + +/// Start the background polling loop. Checks every 60 seconds if any tracker needs polling. +pub fn start(db: Arc>, encryption_key: [u8; 32], http_client: reqwest::Client, app_handle: AppHandle) { + tokio::spawn(async move { + let mut tick = interval(Duration::from_secs(60)); + loop { + tick.tick().await; + poll_all_trackers(&db, &encryption_key, &http_client, &app_handle).await; + } + }); +} + +async fn poll_all_trackers( + db: &Arc>, + encryption_key: &[u8; 32], + http_client: &reqwest::Client, + app_handle: &AppHandle, +) { + let trackers = { + let conn = match db.lock() { + Ok(c) => c, + Err(_) => return, + }; + WatchedTracker::list_all_enabled(&conn).unwrap_or_default() + }; + + if trackers.is_empty() { + return; + } + + // Build Tuleap client from credentials + let client = { + let conn = match db.lock() { + Ok(c) => c, + Err(_) => return, + }; + let cred = match TuleapCredentials::get(&conn) { + Ok(Some(c)) => c, + _ => return, + }; + let password = match crypto::decrypt(encryption_key, &cred.password_encrypted) { + Ok(p) => p, + Err(_) => return, + }; + TuleapClient::new(http_client, &cred.tuleap_url, &cred.username, &password) + }; + + for tracker in &trackers { + if should_poll(tracker) { + poll_single_tracker(db, &client, tracker, app_handle).await; + } + } +} + +fn should_poll(tracker: &WatchedTracker) -> bool { + let Some(last_polled) = &tracker.last_polled_at else { + return true; // Never polled + }; + let Ok(last) = chrono::DateTime::parse_from_rfc3339(last_polled) else { + return true; + }; + let elapsed = chrono::Utc::now().signed_duration_since(last); + elapsed.num_minutes() >= tracker.polling_interval as i64 +} + +async fn poll_single_tracker( + db: &Arc>, + client: &TuleapClient, + tracker: &WatchedTracker, + app_handle: &AppHandle, +) { + let artifacts = match client.get_artifacts(tracker.tracker_id).await { + Ok(a) => a, + Err(e) => { + eprintln!("Poller error for tracker {}: {}", tracker.tracker_label, e); + return; + } + }; + + let filtered = filter_engine::apply_filters(&artifacts, &tracker.filters); + + let mut new_tickets = Vec::new(); + { + let conn = match db.lock() { + Ok(c) => c, + Err(_) => return, + }; + for artifact in &filtered { + let artifact_id = artifact.get("id").and_then(|v| v.as_i64()).unwrap_or(0) as i32; + let artifact_title = artifact.get("title").and_then(|v| v.as_str()).unwrap_or("Untitled"); + let artifact_data = serde_json::to_string(artifact).unwrap_or_default(); + + match ProcessedTicket::insert_if_new(&conn, &tracker.id, artifact_id, artifact_title, &artifact_data) { + Ok(Some(ticket)) => new_tickets.push(ticket), + Ok(None) => {} // already processed + Err(e) => eprintln!("Failed to insert ticket: {}", e), + } + } + let _ = WatchedTracker::update_last_polled(&conn, &tracker.id); + } + + if !new_tickets.is_empty() { + let _ = app_handle.emit("new-tickets-detected", serde_json::json!({ + "tracker_id": tracker.id, + "tracker_label": tracker.tracker_label, + "count": new_tickets.len(), + })); + } +} +``` + +- [x] **Step 2: Spawn poller on app startup** + +In `src-tauri/src/lib.rs`, add the poller startup at the end of the `setup` closure, after `app.manage(...)`: + +```rust + .setup(|app| { + let db_dir = app.path().app_data_dir()?; + std::fs::create_dir_all(&db_dir)?; + + let db_path = db_dir.join("orchai.db"); + let conn = db::init(&db_path).expect("Failed to initialize database"); + + let key_path = db_dir.join("orchai.key"); + let encryption_key = load_or_generate_key(&key_path)?; + + let http_client = reqwest::Client::new(); + + let db_arc = Arc::new(Mutex::new(conn)); + + app.manage(AppState { + db: db_arc.clone(), + encryption_key, + http_client: http_client.clone(), + }); + + // Start background poller + services::poller::start( + db_arc, + encryption_key, + http_client, + app.handle().clone(), + ); + + Ok(()) + }) +``` + +- [x] **Step 3: Verify compilation** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo build +``` + +Expected: compiles. All existing tests still pass. + +- [x] **Step 4: Commit** + +```bash +git add -A +git commit -m "feat: background poller with 60s tick, per-tracker interval, event emission" +``` + +--- + +### Task 10: Frontend types + API wrappers + Settings page + +**Files:** +- Modify: `src/lib/types.ts` +- Modify: `src/lib/api.ts` +- Create: `src/components/settings/SettingsPage.tsx` +- Modify: `src/App.tsx` (add /settings route) +- Modify: `src/components/layout/Sidebar.tsx` (add settings link) + +- [x] **Step 1: Add new TypeScript types** + +Add to `src/lib/types.ts`: + +```typescript +export interface TuleapCredentialsSafe { + id: string; + tuleap_url: string; + username: string; +} + +export interface AgentConfig { + analyst_command: string; + analyst_args: string[]; + developer_command: string; + developer_args: string[]; +} + +export interface Filter { + field: string; + operator: string; + value: string[]; +} + +export interface FilterGroup { + conditions: Filter[]; +} + +export interface TrackerField { + field_id: number; + label: string; + field_type: string; + values: FieldValue[]; +} + +export interface FieldValue { + id: number; + label: string; +} + +export interface WatchedTracker { + id: string; + project_id: string; + tracker_id: number; + tracker_label: string; + polling_interval: number; + agent_config: AgentConfig; + filters: FilterGroup[]; + enabled: boolean; + last_polled_at: string | null; + created_at: string; +} + +export interface ProcessedTicket { + id: string; + tracker_id: string; + artifact_id: number; + artifact_title: string; + artifact_data: string; + status: string; + analyst_report: string | null; + developer_report: string | null; + worktree_path: string | null; + branch_name: string | null; + detected_at: string; + processed_at: string | null; +} +``` + +- [x] **Step 2: Add new API wrappers** + +Add to `src/lib/api.ts`: + +```typescript +import type { + Project, + TuleapCredentialsSafe, + WatchedTracker, + AgentConfig, + FilterGroup, + TrackerField, + ProcessedTicket, +} from "./types"; + +// Credentials +export async function setTuleapCredentials( + tuleapUrl: string, + username: string, + password: string +): Promise { + return invoke("set_tuleap_credentials", { tuleapUrl, username, password }); +} + +export async function getTuleapCredentials(): Promise { + return invoke("get_tuleap_credentials"); +} + +export async function deleteTuleapCredentials(): Promise { + return invoke("delete_tuleap_credentials"); +} + +export async function testTuleapConnection(): Promise { + return invoke("test_tuleap_connection"); +} + +// Trackers +export async function addTracker( + projectId: string, + trackerId: number, + trackerLabel: string, + pollingInterval: number, + agentConfig: AgentConfig, + filters: FilterGroup[] +): Promise { + return invoke("add_tracker", { + projectId, trackerId, trackerLabel, pollingInterval, agentConfig, filters, + }); +} + +export async function listTrackers(projectId: string): Promise { + return invoke("list_trackers", { projectId }); +} + +export async function updateTracker( + id: string, + pollingInterval: number, + agentConfig: AgentConfig, + filters: FilterGroup[], + enabled: boolean +): Promise { + return invoke("update_tracker", { id, pollingInterval, agentConfig, filters, enabled }); +} + +export async function removeTracker(id: string): Promise { + return invoke("remove_tracker", { id }); +} + +export async function getTrackerFields(trackerId: number): Promise { + return invoke("get_tracker_fields", { trackerId }); +} + +// Tickets +export async function listProcessedTickets(projectId: string): Promise { + return invoke("list_processed_tickets", { projectId }); +} + +export async function manualPoll(trackerId: string): Promise { + return invoke("manual_poll", { trackerId }); +} + +export async function getQueueStatus(projectId: string): Promise { + return invoke("get_queue_status", { projectId }); +} +``` + +- [x] **Step 3: Create SettingsPage component** + +Create `src/components/settings/SettingsPage.tsx`: + +```tsx +import { useEffect, useState } from "react"; +import { + getTuleapCredentials, + setTuleapCredentials, + deleteTuleapCredentials, + testTuleapConnection, +} from "../../lib/api"; +import type { TuleapCredentialsSafe } from "../../lib/types"; + +export default function SettingsPage() { + const [credentials, setCredentials] = useState(null); + const [url, setUrl] = useState(""); + const [username, setUsername] = useState(""); + const [password, setPassword] = useState(""); + const [error, setError] = useState(null); + const [success, setSuccess] = useState(null); + const [loading, setLoading] = useState(false); + const [testing, setTesting] = useState(false); + + useEffect(() => { + getTuleapCredentials().then((cred) => { + setCredentials(cred); + if (cred) { + setUrl(cred.tuleap_url); + setUsername(cred.username); + } + }); + }, []); + + async function handleSave(e: React.FormEvent) { + e.preventDefault(); + setError(null); + setSuccess(null); + setLoading(true); + try { + const saved = await setTuleapCredentials(url, username, password); + setCredentials(saved); + setPassword(""); + setSuccess("Credentials saved"); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setLoading(false); + } + } + + async function handleTest() { + setError(null); + setSuccess(null); + setTesting(true); + try { + const msg = await testTuleapConnection(); + setSuccess(msg); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setTesting(false); + } + } + + async function handleDelete() { + if (!window.confirm("Delete Tuleap credentials?")) return; + await deleteTuleapCredentials(); + setCredentials(null); + setUrl(""); + setUsername(""); + setPassword(""); + } + + return ( +
+

Settings

+ +
+

Tuleap Credentials

+ +
+
+ + setUrl(e.target.value)} + required + placeholder="https://tuleap.example.com" + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+
+ + setUsername(e.target.value)} + required + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+
+ + setPassword(e.target.value)} + required={!credentials} + placeholder={credentials ? "••••••••" : ""} + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+ + {error && ( +
{error}
+ )} + {success && ( +
{success}
+ )} + +
+ + {credentials && ( + <> + + + + )} +
+
+
+
+ ); +} +``` + +- [x] **Step 4: Add settings route and sidebar link** + +In `src/App.tsx`, add the settings route: + +```tsx +import SettingsPage from "./components/settings/SettingsPage"; +``` + +Add inside the `}>` block: + +```tsx +} /> +``` + +In `src/components/layout/Sidebar.tsx`, add a settings link at the bottom of the sidebar (before closing ``): + +```tsx +
+ + Settings + +
+``` + +- [x] **Step 5: Verify TypeScript compiles and frontend builds** + +```bash +cd /home/leclere/Projets/orchai +npx tsc --noEmit +npm run build +``` + +Expected: both succeed. + +- [x] **Step 6: Commit** + +```bash +git add -A +git commit -m "feat: frontend types, API wrappers, and Settings page for Tuleap credentials" +``` + +--- + +### Task 11: Tracker config + filter builder UI + +**Files:** +- Create: `src/components/trackers/TrackerList.tsx` +- Create: `src/components/trackers/TrackerConfig.tsx` +- Create: `src/components/trackers/FilterBuilder.tsx` +- Modify: `src/App.tsx` (add tracker routes) + +- [x] **Step 1: Create FilterBuilder component** + +Create `src/components/trackers/FilterBuilder.tsx`: + +```tsx +import type { FilterGroup, Filter, TrackerField } from "../../lib/types"; + +interface Props { + groups: FilterGroup[]; + onChange: (groups: FilterGroup[]) => void; + availableFields: TrackerField[]; +} + +export default function FilterBuilder({ groups, onChange, availableFields }: Props) { + function addGroup() { + onChange([...groups, { conditions: [{ field: "", operator: "In", value: [] }] }]); + } + + function removeGroup(groupIdx: number) { + onChange(groups.filter((_, i) => i !== groupIdx)); + } + + function addCondition(groupIdx: number) { + const updated = [...groups]; + updated[groupIdx] = { + conditions: [...updated[groupIdx].conditions, { field: "", operator: "In", value: [] }], + }; + onChange(updated); + } + + function removeCondition(groupIdx: number, condIdx: number) { + const updated = [...groups]; + updated[groupIdx] = { + conditions: updated[groupIdx].conditions.filter((_, i) => i !== condIdx), + }; + if (updated[groupIdx].conditions.length === 0) { + onChange(updated.filter((_, i) => i !== groupIdx)); + } else { + onChange(updated); + } + } + + function updateCondition(groupIdx: number, condIdx: number, field: keyof Filter, val: string | string[]) { + const updated = [...groups]; + const condition = { ...updated[groupIdx].conditions[condIdx] }; + if (field === "value") { + condition.value = val as string[]; + } else { + condition[field] = val as string; + // Reset value when field changes + if (field === "field") { + condition.value = []; + } + } + updated[groupIdx] = { + conditions: updated[groupIdx].conditions.map((c, i) => (i === condIdx ? condition : c)), + }; + onChange(updated); + } + + function getFieldValues(fieldLabel: string): string[] { + const field = availableFields.find((f) => f.label === fieldLabel); + return field?.values.map((v) => v.label) ?? []; + } + + function toggleValue(groupIdx: number, condIdx: number, val: string) { + const condition = groups[groupIdx].conditions[condIdx]; + const newValues = condition.value.includes(val) + ? condition.value.filter((v) => v !== val) + : [...condition.value, val]; + updateCondition(groupIdx, condIdx, "value", newValues); + } + + return ( +
+ {groups.map((group, groupIdx) => ( +
+
+ + {groupIdx > 0 && "AND "}Group {groupIdx + 1} + + +
+ + {group.conditions.map((condition, condIdx) => ( +
+ {condIdx > 0 && ( + OR + )} + + + + +
+ {condition.field && getFieldValues(condition.field).map((val) => ( + + ))} +
+ + +
+ ))} + + +
+ ))} + + +
+ ); +} +``` + +- [x] **Step 2: Create TrackerConfig component** + +Create `src/components/trackers/TrackerConfig.tsx`: + +```tsx +import { useState, useEffect } from "react"; +import { useNavigate, useParams } from "react-router-dom"; +import { addTracker, getTrackerFields } from "../../lib/api"; +import type { AgentConfig, FilterGroup, TrackerField } from "../../lib/types"; +import FilterBuilder from "./FilterBuilder"; + +export default function TrackerConfig() { + const navigate = useNavigate(); + const { projectId } = useParams(); + + const [trackerId, setTrackerId] = useState(""); + const [trackerLabel, setTrackerLabel] = useState(""); + const [pollingInterval, setPollingInterval] = useState(10); + const [filters, setFilters] = useState([]); + const [agentConfig, setAgentConfig] = useState({ + analyst_command: "claude", + analyst_args: ["--print"], + developer_command: "claude", + developer_args: ["--print"], + }); + const [availableFields, setAvailableFields] = useState([]); + const [error, setError] = useState(null); + const [loading, setLoading] = useState(false); + const [loadingFields, setLoadingFields] = useState(false); + + async function handleLoadFields() { + const id = parseInt(trackerId, 10); + if (isNaN(id)) return; + setLoadingFields(true); + setError(null); + try { + const fields = await getTrackerFields(id); + setAvailableFields(fields); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setLoadingFields(false); + } + } + + async function handleSubmit(e: React.FormEvent) { + e.preventDefault(); + if (!projectId) return; + setError(null); + setLoading(true); + try { + await addTracker( + projectId, + parseInt(trackerId, 10), + trackerLabel, + pollingInterval, + agentConfig, + filters + ); + navigate(`/projects/${projectId}`); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setLoading(false); + } + } + + return ( +
+

Add Tracker

+ +
+
+

Tracker

+
+
+ + setTrackerId(e.target.value)} + required + className="w-full border border-gray-300 rounded px-3 py-2 text-sm" + /> +
+
+ + setTrackerLabel(e.target.value)} + required + placeholder="e.g. Bugs" + className="w-full border border-gray-300 rounded px-3 py-2 text-sm" + /> +
+
+
+ + setPollingInterval(parseInt(e.target.value, 10) || 10)} + min={1} + className="w-32 border border-gray-300 rounded px-3 py-2 text-sm" + /> +
+ +
+ + {availableFields.length > 0 && ( +
+

Filters

+ +
+ )} + +
+

Agent Configuration

+
+
+ + setAgentConfig({ ...agentConfig, analyst_command: e.target.value })} + className="w-full border border-gray-300 rounded px-3 py-2 text-sm" + /> +
+
+ + setAgentConfig({ ...agentConfig, developer_command: e.target.value })} + className="w-full border border-gray-300 rounded px-3 py-2 text-sm" + /> +
+
+
+ + {error && ( +
{error}
+ )} + +
+ + +
+
+
+ ); +} +``` + +- [x] **Step 3: Create TrackerList component** + +Create `src/components/trackers/TrackerList.tsx`: + +```tsx +import { useState } from "react"; +import { Link } from "react-router-dom"; +import { removeTracker, manualPoll, updateTracker } from "../../lib/api"; +import type { WatchedTracker } from "../../lib/types"; + +interface Props { + trackers: WatchedTracker[]; + projectId: string; + onRefresh: () => void; +} + +export default function TrackerList({ trackers, projectId, onRefresh }: Props) { + const [pollingId, setPollingId] = useState(null); + + async function handlePoll(tracker: WatchedTracker) { + setPollingId(tracker.id); + try { + const newTickets = await manualPoll(tracker.id); + if (newTickets.length > 0) { + alert(`${newTickets.length} new ticket(s) detected`); + } else { + alert("No new tickets"); + } + onRefresh(); + } catch (err: unknown) { + alert(err instanceof Error ? err.message : String(err)); + } finally { + setPollingId(null); + } + } + + async function handleToggle(tracker: WatchedTracker) { + await updateTracker( + tracker.id, + tracker.polling_interval, + tracker.agent_config, + tracker.filters, + !tracker.enabled + ); + onRefresh(); + } + + async function handleRemove(tracker: WatchedTracker) { + if (!window.confirm(`Remove tracker "${tracker.tracker_label}"?`)) return; + await removeTracker(tracker.id); + onRefresh(); + } + + if (trackers.length === 0) { + return ( +
+ No trackers configured.{" "} + + Add one + +
+ ); + } + + return ( +
+ {trackers.map((tracker) => ( +
+
+ {tracker.tracker_label} + #{tracker.tracker_id} + + {tracker.enabled ? "Active" : "Paused"} + + {tracker.last_polled_at && ( + + Last poll: {new Date(tracker.last_polled_at).toLocaleTimeString()} + + )} +
+
+ + + +
+
+ ))} + + + Add tracker + +
+ ); +} +``` + +- [x] **Step 4: Add tracker routes in App.tsx** + +Add import: + +```tsx +import TrackerConfig from "./components/trackers/TrackerConfig"; +``` + +Add route inside `}>`: + +```tsx +} /> +``` + +- [x] **Step 5: Verify frontend builds** + +```bash +cd /home/leclere/Projets/orchai +npx tsc --noEmit +npm run build +``` + +- [x] **Step 6: Commit** + +```bash +git add -A +git commit -m "feat: tracker config UI with visual AND/OR filter builder" +``` + +--- + +### Task 12: Update project dashboard + final verification + +**Files:** +- Modify: `src/components/projects/ProjectDashboard.tsx` + +- [x] **Step 1: Update ProjectDashboard to show trackers and recent tickets** + +Replace `src/components/projects/ProjectDashboard.tsx`: + +```tsx +import { useEffect, useState } from "react"; +import { useParams, Link, useNavigate } from "react-router-dom"; +import { getProject, deleteProject, listTrackers, listProcessedTickets } from "../../lib/api"; +import type { Project, WatchedTracker, ProcessedTicket } from "../../lib/types"; +import TrackerList from "../trackers/TrackerList"; + +export default function ProjectDashboard() { + const { projectId } = useParams(); + const navigate = useNavigate(); + const [project, setProject] = useState(null); + const [trackers, setTrackers] = useState([]); + const [tickets, setTickets] = useState([]); + + function loadData() { + if (!projectId) return; + getProject(projectId).then(setProject); + listTrackers(projectId).then(setTrackers); + listProcessedTickets(projectId).then(setTickets); + } + + useEffect(() => { + loadData(); + }, [projectId]); + + async function handleDelete() { + if (!projectId || !project) return; + if (!window.confirm(`Delete project "${project.name}"?`)) return; + await deleteProject(projectId); + window.dispatchEvent(new Event("orchai:refresh-projects")); + navigate("/"); + } + + if (!project) { + return
Loading...
; + } + + return ( +
+
+

{project.name}

+
+ + Edit + + +
+
+ +
+
+ Path: + {project.path} +
+ {project.cloned_from && ( +
+ Cloned from: + {project.cloned_from} +
+ )} +
+ Base branch: + {project.base_branch} +
+
+ +
+

Watched Trackers

+ +
+ + {tickets.length > 0 && ( +
+

Recent Tickets

+
+ {tickets.slice(0, 10).map((ticket) => ( +
+
+ #{ticket.artifact_id} + {ticket.artifact_title} +
+ + {ticket.status} + +
+ ))} +
+
+ )} +
+ ); +} +``` + +- [x] **Step 2: Run all Rust tests** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo test +``` + +Expected: all tests pass (11 Phase 1 + 7 crypto + 5 credential + 7 tuleap_client + 7 tracker + 7 filter + 5 ticket = 49 tests). + +- [x] **Step 3: Run clippy** + +```bash +cd /home/leclere/Projets/orchai/src-tauri +PKG_CONFIG_PATH="/tmp/mypc:$PKG_CONFIG_PATH" cargo clippy -- -D warnings +``` + +Fix any warnings. + +- [x] **Step 4: Verify frontend builds** + +```bash +cd /home/leclere/Projets/orchai +npx tsc --noEmit +npm run build +``` + +- [x] **Step 5: Commit** + +```bash +git add -A +git commit -m "feat: updated project dashboard with tracker list and recent tickets" +``` diff --git a/docs/superpowers/plans/2026-04-13-orchai-phase3-agent-pipeline.md b/docs/superpowers/plans/2026-04-13-orchai-phase3-agent-pipeline.md new file mode 100644 index 0000000..8e74154 --- /dev/null +++ b/docs/superpowers/plans/2026-04-13-orchai-phase3-agent-pipeline.md @@ -0,0 +1,2335 @@ +# Orchai Phase 3: Agent Pipeline Implementation Plan + +> **For agentic workers:** REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (`- [ ]`) syntax for tracking. + +**Goal:** Build the agent orchestrator that consumes pending tickets via a sequential FIFO queue, runs a two-step CLI pipeline (analyst then developer), manages git worktrees for code fixes, and provides a frontend for viewing results (markdown reports + diff). + +**Architecture:** A background tokio task (orchestrator) polls the DB every 10 seconds for Pending tickets. For each ticket it runs an analyst CLI command (e.g. `claude --print`) with a structured prompt via stdin, stores the markdown report, then optionally creates a git worktree and runs a developer CLI command. Tauri events stream progress to the frontend. A Worktree Manager service handles git worktree lifecycle (create, diff, cherry-pick, delete). The frontend adds ticket list/detail pages with markdown rendering and a diff viewer. + +**Tech Stack:** tokio (process spawning, async I/O), git CLI (worktree ops), react-markdown + remark-gfm (report rendering) + +--- + +## Phasing Context + +This is Plan 3 of 4: +- **Plan 1 (done):** Foundation -- Tauri scaffold, SQLite, Project Manager +- **Plan 2 (done):** Tuleap Integration -- credentials, API client, poller, filter engine, tracker config +- **Plan 3 (this):** Agent Pipeline -- orchestrator, worktree manager, ticket processing, results UI +- **Plan 4:** Notifications + Polish -- notifier, system notifications, dashboard + +--- + +## File Structure + +``` +src-tauri/ + Cargo.toml # modify: add tokio process+io-util features + src/ + lib.rs # modify: start orchestrator, register new commands + models/ + mod.rs # modify: add worktree module + ticket.rs # modify: add update methods, list_pending, status constants + worktree.rs # create: Worktree struct + CRUD + services/ + mod.rs # modify: add orchestrator, worktree_manager + worktree_manager.rs # create: git worktree operations + orchestrator.rs # create: queue consumer, CLI runner, prompt builder + commands/ + mod.rs # modify: add orchestrator, worktree + orchestrator.rs # create: retry_ticket, cancel_ticket, get_ticket_result + worktree.rs # create: list_worktrees, get_diff, apply_fix, delete, list_branches + +src/ + lib/ + types.ts # modify: add Worktree type, TicketResult type + api.ts # modify: add orchestrator + worktree API wrappers + components/ + tickets/ + TicketList.tsx # create: filterable ticket table + TicketDetail.tsx # create: info + markdown reports + diff + actions + projects/ + ProjectDashboard.tsx # modify: make ticket items clickable links + App.tsx # modify: add /projects/:projectId/tickets and /tickets/:ticketId routes +``` + +--- + +### Task 1: Add tokio features + extend ProcessedTicket model + +**Files:** +- Modify: `src-tauri/Cargo.toml` +- Modify: `src-tauri/src/models/ticket.rs` + +- [ ] **Step 1: Add tokio process and io-util features to Cargo.toml** + +In `src-tauri/Cargo.toml`, change the tokio line: + +```toml +tokio = { version = "1", features = ["time", "sync", "macros", "process", "io-util"] } +``` + +- [ ] **Step 2: Write failing tests for ProcessedTicket update methods** + +Append these tests to the existing `#[cfg(test)] mod tests` block in `src-tauri/src/models/ticket.rs`: + +```rust + #[test] + fn test_update_status() { + let (conn, tracker_id) = setup(); + let ticket = ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "T1", "{}") + .unwrap() + .unwrap(); + + ProcessedTicket::update_status(&conn, &ticket.id, "Analyzing").unwrap(); + let updated = ProcessedTicket::get_by_id(&conn, &ticket.id).unwrap(); + assert_eq!(updated.status, "Analyzing"); + } + + #[test] + fn test_set_analyst_report() { + let (conn, tracker_id) = setup(); + let ticket = ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "T1", "{}") + .unwrap() + .unwrap(); + + ProcessedTicket::set_analyst_report(&conn, &ticket.id, "## Report\nAll good.").unwrap(); + let updated = ProcessedTicket::get_by_id(&conn, &ticket.id).unwrap(); + assert_eq!(updated.analyst_report.unwrap(), "## Report\nAll good."); + } + + #[test] + fn test_set_developer_report() { + let (conn, tracker_id) = setup(); + let ticket = ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "T1", "{}") + .unwrap() + .unwrap(); + + ProcessedTicket::set_developer_report(&conn, &ticket.id, "Fixed in main.rs").unwrap(); + let updated = ProcessedTicket::get_by_id(&conn, &ticket.id).unwrap(); + assert_eq!(updated.developer_report.unwrap(), "Fixed in main.rs"); + assert!(updated.processed_at.is_some()); + } + + #[test] + fn test_set_worktree_info() { + let (conn, tracker_id) = setup(); + let ticket = ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "T1", "{}") + .unwrap() + .unwrap(); + + ProcessedTicket::set_worktree_info(&conn, &ticket.id, "/tmp/wt", "orchai/1").unwrap(); + let updated = ProcessedTicket::get_by_id(&conn, &ticket.id).unwrap(); + assert_eq!(updated.worktree_path.unwrap(), "/tmp/wt"); + assert_eq!(updated.branch_name.unwrap(), "orchai/1"); + } + + #[test] + fn test_list_pending() { + let (conn, tracker_id) = setup(); + ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "T1", "{}").unwrap(); + ProcessedTicket::insert_if_new(&conn, &tracker_id, 2, "T2", "{}").unwrap(); + + let pending = ProcessedTicket::list_pending(&conn).unwrap(); + assert_eq!(pending.len(), 2); + // FIFO order: T1 first (oldest detected_at) + assert_eq!(pending[0].artifact_id, 1); + assert_eq!(pending[1].artifact_id, 2); + + // Mark one as Analyzing, it should no longer be in pending + ProcessedTicket::update_status(&conn, &pending[0].id, "Analyzing").unwrap(); + let pending2 = ProcessedTicket::list_pending(&conn).unwrap(); + assert_eq!(pending2.len(), 1); + assert_eq!(pending2[0].artifact_id, 2); + } + + #[test] + fn test_set_error() { + let (conn, tracker_id) = setup(); + let ticket = ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "T1", "{}") + .unwrap() + .unwrap(); + + ProcessedTicket::set_error(&conn, &ticket.id, "CLI timeout after 600s").unwrap(); + let updated = ProcessedTicket::get_by_id(&conn, &ticket.id).unwrap(); + assert_eq!(updated.status, "Error"); + assert_eq!(updated.analyst_report.unwrap(), "CLI timeout after 600s"); + } +``` + +- [ ] **Step 3: Run tests to verify they fail** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo test --lib models::ticket::tests -- --nocapture 2>&1 | tail -20` +Expected: compilation errors (methods don't exist yet) + +- [ ] **Step 4: Implement the update methods** + +Add these methods to `impl ProcessedTicket` in `src-tauri/src/models/ticket.rs`, before the closing `}` of the impl block: + +```rust + pub fn update_status(conn: &Connection, id: &str, status: &str) -> Result<()> { + conn.execute( + "UPDATE processed_tickets SET status = ?1 WHERE id = ?2", + params![status, id], + )?; + Ok(()) + } + + pub fn set_analyst_report(conn: &Connection, id: &str, report: &str) -> Result<()> { + conn.execute( + "UPDATE processed_tickets SET analyst_report = ?1 WHERE id = ?2", + params![report, id], + )?; + Ok(()) + } + + pub fn set_developer_report(conn: &Connection, id: &str, report: &str) -> Result<()> { + conn.execute( + "UPDATE processed_tickets SET developer_report = ?1, processed_at = datetime('now') WHERE id = ?2", + params![report, id], + )?; + Ok(()) + } + + pub fn set_worktree_info( + conn: &Connection, + id: &str, + worktree_path: &str, + branch_name: &str, + ) -> Result<()> { + conn.execute( + "UPDATE processed_tickets SET worktree_path = ?1, branch_name = ?2 WHERE id = ?3", + params![worktree_path, branch_name, id], + )?; + Ok(()) + } + + pub fn list_pending(conn: &Connection) -> Result> { + let sql = format!("{} WHERE status = 'Pending' ORDER BY detected_at ASC", SELECT_ALL_COLS); + let mut stmt = conn.prepare(&sql)?; + let rows = stmt.query_map([], from_row)?; + rows.collect() + } + + pub fn set_error(conn: &Connection, id: &str, error_message: &str) -> Result<()> { + conn.execute( + "UPDATE processed_tickets SET status = 'Error', analyst_report = COALESCE(analyst_report, '') || ?1, processed_at = datetime('now') WHERE id = ?2", + params![error_message, id], + )?; + Ok(()) + } +``` + +Also remove `#[allow(dead_code)]` from `list_by_tracker` and `get_by_id` since they will be used now. + +- [ ] **Step 5: Run tests to verify they pass** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo test --lib models::ticket::tests -- --nocapture` +Expected: all ticket tests pass + +- [ ] **Step 6: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src-tauri/Cargo.toml src-tauri/src/models/ticket.rs +git commit -m "feat: extend ProcessedTicket with status updates, list_pending, and set_error" +``` + +--- + +### Task 2: Create Worktree model + +**Files:** +- Create: `src-tauri/src/models/worktree.rs` +- Modify: `src-tauri/src/models/mod.rs` + +- [ ] **Step 1: Add worktree module to mod.rs** + +In `src-tauri/src/models/mod.rs`, add: + +```rust +pub mod worktree; +``` + +- [ ] **Step 2: Create worktree.rs with struct, CRUD, and tests** + +Create `src-tauri/src/models/worktree.rs`: + +```rust +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Worktree { + pub id: String, + pub ticket_id: String, + pub path: String, + pub branch_name: String, + pub status: String, + pub created_at: String, + pub merged_at: Option, + pub merged_into: Option, +} + +fn from_row(row: &rusqlite::Row) -> rusqlite::Result { + Ok(Worktree { + id: row.get(0)?, + ticket_id: row.get(1)?, + path: row.get(2)?, + branch_name: row.get(3)?, + status: row.get(4)?, + created_at: row.get(5)?, + merged_at: row.get(6)?, + merged_into: row.get(7)?, + }) +} + +const SELECT_ALL_COLS: &str = "SELECT id, ticket_id, path, branch_name, status, \ + created_at, merged_at, merged_into FROM worktrees"; + +impl Worktree { + pub fn insert( + conn: &Connection, + ticket_id: &str, + path: &str, + branch_name: &str, + ) -> Result { + let id = Uuid::new_v4().to_string(); + let now = chrono::Utc::now().to_rfc3339(); + + conn.execute( + "INSERT INTO worktrees (id, ticket_id, path, branch_name, status, created_at) \ + VALUES (?1, ?2, ?3, ?4, 'Active', ?5)", + params![id, ticket_id, path, branch_name, now], + )?; + + Ok(Worktree { + id, + ticket_id: ticket_id.to_string(), + path: path.to_string(), + branch_name: branch_name.to_string(), + status: "Active".to_string(), + created_at: now, + merged_at: None, + merged_into: None, + }) + } + + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + let sql = format!("{} WHERE id = ?1", SELECT_ALL_COLS); + conn.query_row(&sql, params![id], from_row) + } + + pub fn get_by_ticket_id(conn: &Connection, ticket_id: &str) -> Result> { + let sql = format!("{} WHERE ticket_id = ?1", SELECT_ALL_COLS); + let mut stmt = conn.prepare(&sql)?; + let mut rows = stmt.query_map(params![ticket_id], from_row)?; + match rows.next() { + Some(Ok(w)) => Ok(Some(w)), + Some(Err(e)) => Err(e), + None => Ok(None), + } + } + + pub fn list_by_project(conn: &Connection, project_id: &str) -> Result> { + let sql = format!( + "SELECT w.id, w.ticket_id, w.path, w.branch_name, w.status, \ + w.created_at, w.merged_at, w.merged_into \ + FROM worktrees w \ + JOIN processed_tickets pt ON w.ticket_id = pt.id \ + JOIN watched_trackers wt ON pt.tracker_id = wt.id \ + WHERE wt.project_id = ?1 \ + ORDER BY w.created_at DESC" + ); + let mut stmt = conn.prepare(&sql)?; + let rows = stmt.query_map(params![project_id], from_row)?; + rows.collect() + } + + pub fn set_merged( + conn: &Connection, + id: &str, + target_branch: &str, + ) -> Result<()> { + conn.execute( + "UPDATE worktrees SET status = 'Merged', merged_at = datetime('now'), merged_into = ?1 WHERE id = ?2", + params![target_branch, id], + )?; + Ok(()) + } + + pub fn delete(conn: &Connection, id: &str) -> Result<()> { + conn.execute("DELETE FROM worktrees WHERE id = ?1", params![id])?; + Ok(()) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + use crate::models::project::Project; + use crate::models::ticket::ProcessedTicket; + use crate::models::tracker::{AgentConfig, WatchedTracker}; + + fn setup() -> (Connection, String) { + let conn = db::init_in_memory().expect("db init"); + let project = Project::insert(&conn, "Test", "/path", None, "main").unwrap(); + let agent_config = AgentConfig { + analyst_command: "echo".into(), + analyst_args: vec![], + developer_command: "echo".into(), + developer_args: vec![], + }; + let tracker = + WatchedTracker::insert(&conn, &project.id, 100, "Bugs", 10, agent_config, vec![]) + .unwrap(); + let ticket = ProcessedTicket::insert_if_new(&conn, &tracker.id, 42, "Bug 42", "{}") + .unwrap() + .unwrap(); + (conn, ticket.id) + } + + #[test] + fn test_insert_and_get_by_id() { + let (conn, ticket_id) = setup(); + + let wt = Worktree::insert(&conn, &ticket_id, "/tmp/orchai-42", "orchai/42").unwrap(); + assert_eq!(wt.status, "Active"); + assert_eq!(wt.branch_name, "orchai/42"); + + let found = Worktree::get_by_id(&conn, &wt.id).unwrap(); + assert_eq!(found.id, wt.id); + assert_eq!(found.ticket_id, ticket_id); + assert_eq!(found.path, "/tmp/orchai-42"); + } + + #[test] + fn test_get_by_ticket_id() { + let (conn, ticket_id) = setup(); + + let none = Worktree::get_by_ticket_id(&conn, &ticket_id).unwrap(); + assert!(none.is_none()); + + Worktree::insert(&conn, &ticket_id, "/tmp/wt", "orchai/42").unwrap(); + + let some = Worktree::get_by_ticket_id(&conn, &ticket_id).unwrap(); + assert!(some.is_some()); + assert_eq!(some.unwrap().ticket_id, ticket_id); + } + + #[test] + fn test_list_by_project() { + let conn = db::init_in_memory().expect("db init"); + let project = Project::insert(&conn, "P1", "/path", None, "main").unwrap(); + let agent_config = AgentConfig { + analyst_command: "echo".into(), + analyst_args: vec![], + developer_command: "echo".into(), + developer_args: vec![], + }; + let tracker = + WatchedTracker::insert(&conn, &project.id, 100, "Bugs", 10, agent_config, vec![]) + .unwrap(); + let t1 = ProcessedTicket::insert_if_new(&conn, &tracker.id, 1, "T1", "{}") + .unwrap() + .unwrap(); + let t2 = ProcessedTicket::insert_if_new(&conn, &tracker.id, 2, "T2", "{}") + .unwrap() + .unwrap(); + + Worktree::insert(&conn, &t1.id, "/wt1", "orchai/1").unwrap(); + Worktree::insert(&conn, &t2.id, "/wt2", "orchai/2").unwrap(); + + let worktrees = Worktree::list_by_project(&conn, &project.id).unwrap(); + assert_eq!(worktrees.len(), 2); + } + + #[test] + fn test_set_merged() { + let (conn, ticket_id) = setup(); + let wt = Worktree::insert(&conn, &ticket_id, "/tmp/wt", "orchai/42").unwrap(); + + Worktree::set_merged(&conn, &wt.id, "feature/login").unwrap(); + let updated = Worktree::get_by_id(&conn, &wt.id).unwrap(); + assert_eq!(updated.status, "Merged"); + assert_eq!(updated.merged_into.unwrap(), "feature/login"); + assert!(updated.merged_at.is_some()); + } + + #[test] + fn test_delete() { + let (conn, ticket_id) = setup(); + let wt = Worktree::insert(&conn, &ticket_id, "/tmp/wt", "orchai/42").unwrap(); + + Worktree::delete(&conn, &wt.id).unwrap(); + let result = Worktree::get_by_id(&conn, &wt.id); + assert!(result.is_err()); // Not found + } +} +``` + +- [ ] **Step 3: Run tests to verify they pass** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo test --lib models::worktree::tests -- --nocapture` +Expected: all 5 worktree tests pass + +- [ ] **Step 4: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src-tauri/src/models/mod.rs src-tauri/src/models/worktree.rs +git commit -m "feat: add Worktree model with CRUD operations" +``` + +--- + +### Task 3: Worktree Manager service + +**Files:** +- Create: `src-tauri/src/services/worktree_manager.rs` +- Modify: `src-tauri/src/services/mod.rs` + +- [ ] **Step 1: Add worktree_manager to services/mod.rs** + +In `src-tauri/src/services/mod.rs`, add: + +```rust +pub mod worktree_manager; +``` + +- [ ] **Step 2: Create worktree_manager.rs with git operations** + +Create `src-tauri/src/services/worktree_manager.rs`: + +```rust +use std::path::Path; +use std::process::Command; + +fn run_git(project_path: &str, args: &[&str]) -> Result { + let output = Command::new("git") + .args(args) + .current_dir(project_path) + .output() + .map_err(|e| format!("Failed to run git: {}", e))?; + + if output.status.success() { + Ok(String::from_utf8_lossy(&output.stdout).to_string()) + } else { + let stderr = String::from_utf8_lossy(&output.stderr); + Err(format!("git {} failed: {}", args.join(" "), stderr)) + } +} + +/// Creates a git worktree at `.orchai/worktrees/orchai-{artifact_id}` +/// with a new branch `orchai/{artifact_id}` based on `base_branch`. +/// Returns (worktree_path, branch_name). +pub fn create_worktree( + project_path: &str, + base_branch: &str, + artifact_id: i32, +) -> Result<(String, String), String> { + let orchai_dir = Path::new(project_path).join(".orchai").join("worktrees"); + std::fs::create_dir_all(&orchai_dir) + .map_err(|e| format!("Failed to create .orchai/worktrees dir: {}", e))?; + + let worktree_name = format!("orchai-{}", artifact_id); + let worktree_path = orchai_dir.join(&worktree_name); + let branch_name = format!("orchai/{}", artifact_id); + + let wt_path_str = worktree_path + .to_str() + .ok_or("Invalid worktree path")?; + + run_git( + project_path, + &["worktree", "add", wt_path_str, "-b", &branch_name, base_branch], + )?; + + Ok((wt_path_str.to_string(), branch_name)) +} + +/// Returns the unified diff between the base branch and the worktree branch. +pub fn get_diff(project_path: &str, base_branch: &str, branch_name: &str) -> Result { + let range = format!("{}...{}", base_branch, branch_name); + run_git(project_path, &["diff", &range]) +} + +/// Lists commit hashes on the worktree branch that are not on the base branch (oldest first). +pub fn list_commits( + project_path: &str, + base_branch: &str, + branch_name: &str, +) -> Result, String> { + let range = format!("{}..{}", base_branch, branch_name); + let output = run_git(project_path, &["log", &range, "--format=%H", "--reverse"])?; + Ok(output.lines().filter(|l| !l.is_empty()).map(String::from).collect()) +} + +/// Cherry-picks commits from the worktree branch into the target branch. +/// Saves and restores the current branch. +pub fn apply_fix( + project_path: &str, + base_branch: &str, + branch_name: &str, + target_branch: &str, +) -> Result<(), String> { + let commits = list_commits(project_path, base_branch, branch_name)?; + if commits.is_empty() { + return Err("No commits to cherry-pick".to_string()); + } + + // Save current branch + let current = run_git(project_path, &["rev-parse", "--abbrev-ref", "HEAD"])?; + let current = current.trim(); + + // Checkout target branch + run_git(project_path, &["checkout", target_branch])?; + + // Cherry-pick each commit + let mut cherry_args = vec!["cherry-pick"]; + let commit_refs: Vec<&str> = commits.iter().map(|s| s.as_str()).collect(); + cherry_args.extend(&commit_refs); + + let result = run_git(project_path, &cherry_args); + + if let Err(e) = &result { + // Abort cherry-pick on conflict + let _ = run_git(project_path, &["cherry-pick", "--abort"]); + // Restore original branch + let _ = run_git(project_path, &["checkout", current]); + return Err(format!("Cherry-pick failed (conflict?): {}", e)); + } + + // Restore original branch + run_git(project_path, &["checkout", current])?; + + Ok(()) +} + +/// Removes the git worktree and deletes the local branch. +pub fn delete_worktree( + project_path: &str, + worktree_path: &str, + branch_name: &str, +) -> Result<(), String> { + // Remove worktree (force in case of unclean state) + run_git(project_path, &["worktree", "remove", worktree_path, "--force"])?; + // Delete branch + let _ = run_git(project_path, &["branch", "-D", branch_name]); + Ok(()) +} + +/// Lists local branch names. +pub fn list_local_branches(project_path: &str) -> Result, String> { + let output = run_git(project_path, &["branch", "--format=%(refname:short)"])?; + Ok(output.lines().filter(|l| !l.is_empty()).map(String::from).collect()) +} + +#[cfg(test)] +mod tests { + use super::*; + use std::process::Command; + + fn setup_test_repo() -> tempfile::TempDir { + let dir = tempfile::tempdir().expect("create temp dir"); + let path = dir.path().to_str().unwrap(); + + Command::new("git").args(["init"]).current_dir(path).output().unwrap(); + Command::new("git").args(["config", "user.email", "test@test.com"]).current_dir(path).output().unwrap(); + Command::new("git").args(["config", "user.name", "Test"]).current_dir(path).output().unwrap(); + + // Create initial commit on main + std::fs::write(dir.path().join("README.md"), "# Test").unwrap(); + Command::new("git").args(["add", "."]).current_dir(path).output().unwrap(); + Command::new("git").args(["commit", "-m", "init"]).current_dir(path).output().unwrap(); + + dir + } + + #[test] + fn test_create_worktree() { + let dir = setup_test_repo(); + let path = dir.path().to_str().unwrap(); + + let (wt_path, branch) = create_worktree(path, "main", 42).unwrap(); + assert!(wt_path.contains("orchai-42")); + assert_eq!(branch, "orchai/42"); + assert!(Path::new(&wt_path).exists()); + } + + #[test] + fn test_get_diff_empty() { + let dir = setup_test_repo(); + let path = dir.path().to_str().unwrap(); + + let (_, branch) = create_worktree(path, "main", 1).unwrap(); + let diff = get_diff(path, "main", &branch).unwrap(); + assert!(diff.is_empty(), "No changes yet, diff should be empty"); + } + + #[test] + fn test_get_diff_with_changes() { + let dir = setup_test_repo(); + let path = dir.path().to_str().unwrap(); + + let (wt_path, branch) = create_worktree(path, "main", 2).unwrap(); + + // Make a change in the worktree + std::fs::write(Path::new(&wt_path).join("fix.txt"), "fixed").unwrap(); + Command::new("git").args(["add", "."]).current_dir(&wt_path).output().unwrap(); + Command::new("git").args(["commit", "-m", "fix"]).current_dir(&wt_path).output().unwrap(); + + let diff = get_diff(path, "main", &branch).unwrap(); + assert!(diff.contains("fix.txt")); + assert!(diff.contains("+fixed")); + } + + #[test] + fn test_list_commits() { + let dir = setup_test_repo(); + let path = dir.path().to_str().unwrap(); + + let (wt_path, branch) = create_worktree(path, "main", 3).unwrap(); + + // Make two commits in the worktree + std::fs::write(Path::new(&wt_path).join("a.txt"), "a").unwrap(); + Command::new("git").args(["add", "."]).current_dir(&wt_path).output().unwrap(); + Command::new("git").args(["commit", "-m", "first"]).current_dir(&wt_path).output().unwrap(); + + std::fs::write(Path::new(&wt_path).join("b.txt"), "b").unwrap(); + Command::new("git").args(["add", "."]).current_dir(&wt_path).output().unwrap(); + Command::new("git").args(["commit", "-m", "second"]).current_dir(&wt_path).output().unwrap(); + + let commits = list_commits(path, "main", &branch).unwrap(); + assert_eq!(commits.len(), 2); + } + + #[test] + fn test_list_local_branches() { + let dir = setup_test_repo(); + let path = dir.path().to_str().unwrap(); + + create_worktree(path, "main", 10).unwrap(); + let branches = list_local_branches(path).unwrap(); + assert!(branches.contains(&"main".to_string())); + assert!(branches.contains(&"orchai/10".to_string())); + } + + #[test] + fn test_delete_worktree() { + let dir = setup_test_repo(); + let path = dir.path().to_str().unwrap(); + + let (wt_path, branch) = create_worktree(path, "main", 99).unwrap(); + assert!(Path::new(&wt_path).exists()); + + delete_worktree(path, &wt_path, &branch).unwrap(); + assert!(!Path::new(&wt_path).exists()); + + let branches = list_local_branches(path).unwrap(); + assert!(!branches.contains(&"orchai/99".to_string())); + } + + #[test] + fn test_apply_fix() { + let dir = setup_test_repo(); + let path = dir.path().to_str().unwrap(); + + // Create a target branch + Command::new("git").args(["branch", "feature/test"]).current_dir(path).output().unwrap(); + + // Create worktree and make a change + let (wt_path, branch) = create_worktree(path, "main", 7).unwrap(); + std::fs::write(Path::new(&wt_path).join("fix.txt"), "the fix").unwrap(); + Command::new("git").args(["add", "."]).current_dir(&wt_path).output().unwrap(); + Command::new("git").args(["commit", "-m", "apply fix"]).current_dir(&wt_path).output().unwrap(); + + // Apply fix to feature branch + apply_fix(path, "main", &branch, "feature/test").unwrap(); + + // Verify: check out feature branch and verify the file exists + Command::new("git").args(["checkout", "feature/test"]).current_dir(path).output().unwrap(); + assert!(Path::new(path).join("fix.txt").exists()); + + // Go back to main + Command::new("git").args(["checkout", "main"]).current_dir(path).output().unwrap(); + } +} +``` + +- [ ] **Step 3: Add `tempfile` dev-dependency to Cargo.toml** + +In `src-tauri/Cargo.toml`, add under `[dev-dependencies]`: + +```toml +[dev-dependencies] +tempfile = "3" +``` + +- [ ] **Step 4: Run tests to verify they pass** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo test --lib services::worktree_manager::tests -- --nocapture` +Expected: all 7 worktree manager tests pass + +- [ ] **Step 5: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src-tauri/Cargo.toml src-tauri/src/services/mod.rs src-tauri/src/services/worktree_manager.rs +git commit -m "feat: add Worktree Manager service with git worktree operations" +``` + +--- + +### Task 4: Agent Orchestrator - prompt building and verdict parsing + +**Files:** +- Create: `src-tauri/src/services/orchestrator.rs` +- Modify: `src-tauri/src/services/mod.rs` + +- [ ] **Step 1: Add orchestrator to services/mod.rs** + +In `src-tauri/src/services/mod.rs`, add: + +```rust +pub mod orchestrator; +``` + +- [ ] **Step 2: Create orchestrator.rs with prompt building, verdict parsing, and tests** + +Create `src-tauri/src/services/orchestrator.rs`: + +```rust +use crate::models::project::Project; +use crate::models::ticket::ProcessedTicket; +use crate::models::tracker::WatchedTracker; +use crate::models::worktree::Worktree; +use crate::services::worktree_manager; +use rusqlite::Connection; +use std::sync::{Arc, Mutex}; +use tauri::{AppHandle, Emitter}; +use tokio::io::{AsyncBufReadExt, BufReader}; +use tokio::process::Command; +use tokio::time::{interval, timeout, Duration}; + +#[derive(Debug, Clone, PartialEq)] +pub enum Verdict { + FixNeeded, + NoFix, +} + +pub fn build_analyst_prompt(ticket: &ProcessedTicket, project: &Project) -> String { + format!( + r#"Tu es un analyste technique. Voici un ticket Tuleap a analyser. + +## Ticket +- ID: {artifact_id} +- Titre: {title} +- Donnees: {data} + +## Contexte +- Projet: {project_name} +- Repo: {project_path} +- Branche de base: {base_branch} + +## Ta mission +1. Analyse le ticket et identifie les fichiers/fonctions concernes +2. Explique techniquement le probleme +3. Evalue si une correction de code est necessaire +4. Produis un rapport structure en markdown + +Termine ton rapport par un de ces verdicts sur une ligne separee: +[VERDICT: FIX_NEEDED] si une correction de code est necessaire +[VERDICT: NO_FIX] si aucune correction n'est necessaire"#, + artifact_id = ticket.artifact_id, + title = ticket.artifact_title, + data = ticket.artifact_data, + project_name = project.name, + project_path = project.path, + base_branch = project.base_branch, + ) +} + +pub fn build_developer_prompt( + ticket: &ProcessedTicket, + project: &Project, + analyst_report: &str, + worktree_path: &str, +) -> String { + format!( + r#"Tu es un developpeur. Tu dois corriger un bug ou implementer une fonctionnalite d'apres l'analyse suivante. + +## Rapport d'analyse +{analyst_report} + +## Ticket +- ID: {artifact_id} +- Titre: {title} + +## Contexte +- Projet: {project_name} +- Repo (worktree): {worktree_path} +- Branche de base: {base_branch} + +## Ta mission +1. Implemente la correction dans le code +2. Fais des commits atomiques avec des messages clairs +3. Produis un rapport en markdown decrivant les changements effectues"#, + analyst_report = analyst_report, + artifact_id = ticket.artifact_id, + title = ticket.artifact_title, + project_name = project.name, + worktree_path = worktree_path, + base_branch = project.base_branch, + ) +} + +pub fn parse_verdict(report: &str) -> Verdict { + // Search from the end of the report for the verdict line + for line in report.lines().rev() { + let trimmed = line.trim(); + if trimmed.contains("[VERDICT: NO_FIX]") { + return Verdict::NoFix; + } + if trimmed.contains("[VERDICT: FIX_NEEDED]") { + return Verdict::FixNeeded; + } + } + // Default: assume fix is needed if no verdict found + Verdict::FixNeeded +} + +/// Runs a CLI command with the prompt piped to stdin. +/// Streams stdout lines as Tauri events. Returns the full stdout output. +pub async fn run_cli_command( + command: &str, + args: &[String], + prompt: &str, + working_dir: &str, + timeout_secs: u64, + app_handle: &AppHandle, + ticket_id: &str, +) -> Result { + let mut child = Command::new(command) + .args(args) + .stdin(std::process::Stdio::piped()) + .stdout(std::process::Stdio::piped()) + .stderr(std::process::Stdio::piped()) + .current_dir(working_dir) + .spawn() + .map_err(|e| format!("Failed to spawn '{}': {}", command, e))?; + + // Write prompt to stdin + if let Some(mut stdin) = child.stdin.take() { + use tokio::io::AsyncWriteExt; + stdin + .write_all(prompt.as_bytes()) + .await + .map_err(|e| format!("Failed to write to stdin: {}", e))?; + // stdin is dropped here, closing it + } + + // Read stdout line by line, streaming events + let stdout = child.stdout.take().ok_or("Failed to capture stdout")?; + let mut reader = BufReader::new(stdout).lines(); + let mut output = String::new(); + + let read_future = async { + while let Ok(Some(line)) = reader.next_line().await { + let _ = app_handle.emit( + "ticket-processing-progress", + serde_json::json!({ + "ticket_id": ticket_id, + "output_chunk": line, + }), + ); + output.push_str(&line); + output.push('\n'); + } + output + }; + + let result = timeout(Duration::from_secs(timeout_secs), read_future) + .await + .map_err(|_| format!("CLI command timed out after {}s", timeout_secs))?; + + // Wait for process to finish + let status = child + .wait() + .await + .map_err(|e| format!("Failed to wait for process: {}", e))?; + + if !status.success() { + let code = status.code().unwrap_or(-1); + return Err(format!("CLI command exited with code {}", code)); + } + + Ok(result) +} + +/// Processes a single ticket through the analyst -> developer pipeline. +async fn process_ticket( + db: &Arc>, + app_handle: &AppHandle, +) -> Result { + // 1. Get next pending ticket + its tracker + project (under lock) + let (ticket, tracker, project) = { + let conn = db.lock().map_err(|e| format!("DB lock failed: {}", e))?; + + let pending = ProcessedTicket::list_pending(&conn) + .map_err(|e| format!("list_pending failed: {}", e))?; + + let ticket = match pending.into_iter().next() { + Some(t) => t, + None => return Ok(false), // No pending tickets + }; + + let tracker = WatchedTracker::get_by_id(&conn, &ticket.tracker_id) + .map_err(|e| format!("get tracker failed: {}", e))?; + + let project = Project::get_by_id(&conn, &tracker.project_id) + .map_err(|e| format!("get project failed: {}", e))?; + + // Mark as Analyzing before releasing lock + ProcessedTicket::update_status(&conn, &ticket.id, "Analyzing") + .map_err(|e| format!("update_status failed: {}", e))?; + + (ticket, tracker, project) + }; // lock released + + // Emit start event + let _ = app_handle.emit( + "ticket-processing-started", + serde_json::json!({ + "ticket_id": ticket.id, + "step": "analyst", + }), + ); + + // 2. Run analyst + let analyst_prompt = build_analyst_prompt(&ticket, &project); + let analyst_result = run_cli_command( + &tracker.agent_config.analyst_command, + &tracker.agent_config.analyst_args, + &analyst_prompt, + &project.path, + 600, // 10 minute timeout + app_handle, + &ticket.id, + ) + .await; + + let analyst_report = match analyst_result { + Ok(report) => report, + Err(e) => { + let conn = db.lock().map_err(|e| format!("DB lock: {}", e))?; + let _ = ProcessedTicket::set_error(&conn, &ticket.id, &e); + let _ = app_handle.emit( + "ticket-processing-error", + serde_json::json!({ "ticket_id": ticket.id, "error": e }), + ); + return Ok(true); // Processed (with error), continue to next + } + }; + + // 3. Store analyst report + { + let conn = db.lock().map_err(|e| format!("DB lock: {}", e))?; + ProcessedTicket::set_analyst_report(&conn, &ticket.id, &analyst_report) + .map_err(|e| format!("set_analyst_report: {}", e))?; + } + + // 4. Check verdict + let verdict = parse_verdict(&analyst_report); + if verdict == Verdict::NoFix { + let conn = db.lock().map_err(|e| format!("DB lock: {}", e))?; + ProcessedTicket::update_status(&conn, &ticket.id, "Done") + .map_err(|e| format!("update_status: {}", e))?; + let _ = app_handle.emit( + "ticket-processing-done", + serde_json::json!({ "ticket_id": ticket.id }), + ); + return Ok(true); + } + + // 5. Check if ticket was cancelled while analyst was running + { + let conn = db.lock().map_err(|e| format!("DB lock: {}", e))?; + let current = ProcessedTicket::get_by_id(&conn, &ticket.id) + .map_err(|e| format!("get_by_id: {}", e))?; + if current.status == "Cancelled" { + return Ok(true); + } + } + + // 6. Create worktree + let (wt_path, branch_name) = + worktree_manager::create_worktree(&project.path, &project.base_branch, ticket.artifact_id) + .map_err(|e| { + let conn = db.lock().ok(); + if let Some(conn) = conn { + let _ = ProcessedTicket::set_error(&conn, &ticket.id, &e); + } + e + })?; + + // Store worktree info in DB + { + let conn = db.lock().map_err(|e| format!("DB lock: {}", e))?; + ProcessedTicket::set_worktree_info(&conn, &ticket.id, &wt_path, &branch_name) + .map_err(|e| format!("set_worktree_info: {}", e))?; + Worktree::insert(&conn, &ticket.id, &wt_path, &branch_name) + .map_err(|e| format!("insert worktree: {}", e))?; + ProcessedTicket::update_status(&conn, &ticket.id, "Developing") + .map_err(|e| format!("update_status: {}", e))?; + } + + // Emit developer start + let _ = app_handle.emit( + "ticket-processing-started", + serde_json::json!({ + "ticket_id": ticket.id, + "step": "developer", + }), + ); + + // 7. Run developer + let developer_prompt = + build_developer_prompt(&ticket, &project, &analyst_report, &wt_path); + let developer_result = run_cli_command( + &tracker.agent_config.developer_command, + &tracker.agent_config.developer_args, + &developer_prompt, + &wt_path, + 600, + app_handle, + &ticket.id, + ) + .await; + + let developer_report = match developer_result { + Ok(report) => report, + Err(e) => { + let conn = db.lock().map_err(|e2| format!("DB lock: {}", e2))?; + let _ = ProcessedTicket::set_error(&conn, &ticket.id, &e); + let _ = app_handle.emit( + "ticket-processing-error", + serde_json::json!({ "ticket_id": ticket.id, "error": e }), + ); + return Ok(true); + } + }; + + // 8. Store developer report and mark done + { + let conn = db.lock().map_err(|e| format!("DB lock: {}", e))?; + ProcessedTicket::set_developer_report(&conn, &ticket.id, &developer_report) + .map_err(|e| format!("set_developer_report: {}", e))?; + ProcessedTicket::update_status(&conn, &ticket.id, "Done") + .map_err(|e| format!("update_status: {}", e))?; + } + + let _ = app_handle.emit( + "ticket-processing-done", + serde_json::json!({ "ticket_id": ticket.id }), + ); + + Ok(true) // Processed a ticket +} + +/// Starts the orchestrator background task that consumes the ticket queue. +pub fn start(db: Arc>, app_handle: AppHandle) { + tokio::spawn(async move { + let mut tick = interval(Duration::from_secs(10)); + loop { + tick.tick().await; + match process_ticket(&db, &app_handle).await { + Ok(true) => { + // Processed a ticket, immediately check for more + continue; + } + Ok(false) => { + // No pending tickets, wait for next tick + } + Err(e) => { + eprintln!("orchestrator: {}", e); + } + } + } + }); +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_build_analyst_prompt_contains_ticket_info() { + let ticket = ProcessedTicket { + id: "t1".into(), + tracker_id: "tr1".into(), + artifact_id: 42, + artifact_title: "Login crash on empty password".into(), + artifact_data: r#"{"id":42,"title":"Login crash"}"#.into(), + status: "Pending".into(), + analyst_report: None, + developer_report: None, + worktree_path: None, + branch_name: None, + detected_at: "2026-01-01T00:00:00Z".into(), + processed_at: None, + }; + let project = Project { + id: "p1".into(), + name: "MyApp".into(), + path: "/home/user/myapp".into(), + cloned_from: None, + base_branch: "stable".into(), + created_at: "2026-01-01T00:00:00Z".into(), + }; + + let prompt = build_analyst_prompt(&ticket, &project); + assert!(prompt.contains("42")); + assert!(prompt.contains("Login crash on empty password")); + assert!(prompt.contains("MyApp")); + assert!(prompt.contains("/home/user/myapp")); + assert!(prompt.contains("stable")); + assert!(prompt.contains("[VERDICT: FIX_NEEDED]")); + assert!(prompt.contains("[VERDICT: NO_FIX]")); + } + + #[test] + fn test_build_developer_prompt_contains_report() { + let ticket = ProcessedTicket { + id: "t1".into(), + tracker_id: "tr1".into(), + artifact_id: 42, + artifact_title: "Login crash".into(), + artifact_data: "{}".into(), + status: "Developing".into(), + analyst_report: None, + developer_report: None, + worktree_path: None, + branch_name: None, + detected_at: "2026-01-01T00:00:00Z".into(), + processed_at: None, + }; + let project = Project { + id: "p1".into(), + name: "MyApp".into(), + path: "/home/user/myapp".into(), + cloned_from: None, + base_branch: "main".into(), + created_at: "2026-01-01T00:00:00Z".into(), + }; + + let prompt = build_developer_prompt(&ticket, &project, "## Bug found in auth.rs", "/tmp/wt"); + assert!(prompt.contains("## Bug found in auth.rs")); + assert!(prompt.contains("42")); + assert!(prompt.contains("/tmp/wt")); + } + + #[test] + fn test_parse_verdict_fix_needed() { + let report = "## Analysis\nBug found.\n[VERDICT: FIX_NEEDED]\n"; + assert_eq!(parse_verdict(report), Verdict::FixNeeded); + } + + #[test] + fn test_parse_verdict_no_fix() { + let report = "## Analysis\nThis is a feature request, not a bug.\n[VERDICT: NO_FIX]\n"; + assert_eq!(parse_verdict(report), Verdict::NoFix); + } + + #[test] + fn test_parse_verdict_missing_defaults_to_fix() { + let report = "## Analysis\nSomething is wrong but I forgot the verdict."; + assert_eq!(parse_verdict(report), Verdict::FixNeeded); + } + + #[test] + fn test_parse_verdict_embedded_in_line() { + let report = "Verdict: [VERDICT: NO_FIX] - no code change needed."; + assert_eq!(parse_verdict(report), Verdict::NoFix); + } +} +``` + +- [ ] **Step 3: Run tests to verify they pass** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo test --lib services::orchestrator::tests -- --nocapture` +Expected: all 6 orchestrator tests pass + +- [ ] **Step 4: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src-tauri/src/services/mod.rs src-tauri/src/services/orchestrator.rs +git commit -m "feat: add Agent Orchestrator with prompt building, verdict parsing, and CLI pipeline" +``` + +--- + +### Task 5: Tauri commands for orchestrator and worktree + +**Files:** +- Create: `src-tauri/src/commands/orchestrator.rs` +- Create: `src-tauri/src/commands/worktree.rs` +- Modify: `src-tauri/src/commands/mod.rs` + +- [ ] **Step 1: Add new command modules to commands/mod.rs** + +In `src-tauri/src/commands/mod.rs`, add: + +```rust +pub mod orchestrator; +pub mod worktree; +``` + +- [ ] **Step 2: Create commands/orchestrator.rs** + +Create `src-tauri/src/commands/orchestrator.rs`: + +```rust +use crate::error::AppError; +use crate::models::ticket::ProcessedTicket; +use crate::models::worktree::Worktree; +use crate::AppState; +use serde::Serialize; +use tauri::State; + +#[derive(Debug, Clone, Serialize)] +pub struct TicketResult { + pub ticket: ProcessedTicket, + pub worktree: Option, +} + +#[tauri::command] +pub fn get_ticket_result( + state: State<'_, AppState>, + ticket_id: String, +) -> Result { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + let ticket = ProcessedTicket::get_by_id(&conn, &ticket_id)?; + let worktree = Worktree::get_by_ticket_id(&conn, &ticket_id)?; + Ok(TicketResult { ticket, worktree }) +} + +#[tauri::command] +pub fn retry_ticket( + state: State<'_, AppState>, + ticket_id: String, +) -> Result<(), AppError> { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + let ticket = ProcessedTicket::get_by_id(&conn, &ticket_id)?; + + // Only allow retry for Error or Done tickets + if ticket.status != "Error" && ticket.status != "Done" && ticket.status != "Cancelled" { + return Err(AppError::from(format!( + "Cannot retry ticket with status '{}'", + ticket.status + ))); + } + + // Reset to Pending + ProcessedTicket::update_status(&conn, &ticket_id, "Pending")?; + // Clear reports + conn.execute( + "UPDATE processed_tickets SET analyst_report = NULL, developer_report = NULL, \ + worktree_path = NULL, branch_name = NULL, processed_at = NULL WHERE id = ?1", + rusqlite::params![ticket_id], + )?; + + // Clean up worktree if exists + if let Some(wt) = Worktree::get_by_ticket_id(&conn, &ticket_id)? { + if wt.status == "Active" { + // Best effort: delete the worktree on disk + let project_id = { + let tracker = crate::models::tracker::WatchedTracker::get_by_id(&conn, &ticket.tracker_id)?; + tracker.project_id + }; + let project = crate::models::project::Project::get_by_id(&conn, &project_id)?; + let _ = crate::services::worktree_manager::delete_worktree( + &project.path, + &wt.path, + &wt.branch_name, + ); + } + Worktree::delete(&conn, &wt.id)?; + } + + Ok(()) +} + +#[tauri::command] +pub fn cancel_ticket( + state: State<'_, AppState>, + ticket_id: String, +) -> Result<(), AppError> { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + let ticket = ProcessedTicket::get_by_id(&conn, &ticket_id)?; + + if ticket.status == "Done" || ticket.status == "Cancelled" { + return Err(AppError::from(format!( + "Cannot cancel ticket with status '{}'", + ticket.status + ))); + } + + ProcessedTicket::update_status(&conn, &ticket_id, "Cancelled")?; + Ok(()) +} +``` + +- [ ] **Step 3: Create commands/worktree.rs** + +Create `src-tauri/src/commands/worktree.rs`: + +```rust +use crate::error::AppError; +use crate::models::project::Project; +use crate::models::ticket::ProcessedTicket; +use crate::models::tracker::WatchedTracker; +use crate::models::worktree::Worktree; +use crate::services::worktree_manager; +use crate::AppState; +use tauri::State; + +#[tauri::command] +pub fn list_worktrees( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + let worktrees = Worktree::list_by_project(&conn, &project_id)?; + Ok(worktrees) +} + +#[tauri::command] +pub fn get_worktree_diff( + state: State<'_, AppState>, + worktree_id: String, +) -> Result { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + + let wt = Worktree::get_by_id(&conn, &worktree_id)?; + let ticket = ProcessedTicket::get_by_id(&conn, &wt.ticket_id)?; + let tracker = WatchedTracker::get_by_id(&conn, &ticket.tracker_id)?; + let project = Project::get_by_id(&conn, &tracker.project_id)?; + + drop(conn); // Release lock before git operations + + let diff = worktree_manager::get_diff(&project.path, &project.base_branch, &wt.branch_name) + .map_err(AppError::from)?; + + Ok(diff) +} + +#[tauri::command] +pub fn apply_fix_to_branch( + state: State<'_, AppState>, + worktree_id: String, + target_branch: String, +) -> Result<(), AppError> { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + + let wt = Worktree::get_by_id(&conn, &worktree_id)?; + let ticket = ProcessedTicket::get_by_id(&conn, &wt.ticket_id)?; + let tracker = WatchedTracker::get_by_id(&conn, &ticket.tracker_id)?; + let project = Project::get_by_id(&conn, &tracker.project_id)?; + + drop(conn); // Release lock before git operations + + worktree_manager::apply_fix( + &project.path, + &project.base_branch, + &wt.branch_name, + &target_branch, + ) + .map_err(AppError::from)?; + + // Mark worktree as merged + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + Worktree::set_merged(&conn, &worktree_id, &target_branch)?; + + Ok(()) +} + +#[tauri::command] +pub fn delete_worktree_cmd( + state: State<'_, AppState>, + worktree_id: String, +) -> Result<(), AppError> { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + + let wt = Worktree::get_by_id(&conn, &worktree_id)?; + let ticket = ProcessedTicket::get_by_id(&conn, &wt.ticket_id)?; + let tracker = WatchedTracker::get_by_id(&conn, &ticket.tracker_id)?; + let project = Project::get_by_id(&conn, &tracker.project_id)?; + + drop(conn); // Release lock before git operations + + worktree_manager::delete_worktree(&project.path, &wt.path, &wt.branch_name) + .map_err(AppError::from)?; + + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + Worktree::delete(&conn, &worktree_id)?; + + Ok(()) +} + +#[tauri::command] +pub fn list_local_branches( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let conn = state.db.lock().map_err(|e| AppError::from(e.to_string()))?; + let project = Project::get_by_id(&conn, &project_id)?; + + drop(conn); + + let branches = + worktree_manager::list_local_branches(&project.path).map_err(AppError::from)?; + Ok(branches) +} +``` + +- [ ] **Step 4: Verify compilation** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo check 2>&1 | tail -10` +Expected: compiles without errors + +- [ ] **Step 5: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src-tauri/src/commands/mod.rs src-tauri/src/commands/orchestrator.rs src-tauri/src/commands/worktree.rs +git commit -m "feat: add Tauri commands for orchestrator (retry, cancel, result) and worktree (diff, apply, delete, branches)" +``` + +--- + +### Task 6: Wire orchestrator into app startup and register commands + +**Files:** +- Modify: `src-tauri/src/lib.rs` + +- [ ] **Step 1: Start orchestrator in setup and register new commands** + +In `src-tauri/src/lib.rs`, modify the `setup` closure to start the orchestrator after the poller, and add all new commands to the invoke_handler. + +The setup block should clone db_arc a second time for the orchestrator: + +```rust + // Start background poller + services::poller::start( + db_arc.clone(), + encryption_key, + http_client, + app.handle().clone(), + ); + + // Start agent orchestrator + services::orchestrator::start( + db_arc, + app.handle().clone(), + ); +``` + +Add the new commands to `invoke_handler`: + +```rust + .invoke_handler(tauri::generate_handler![ + commands::project::create_project, + commands::project::list_projects, + commands::project::get_project, + commands::project::update_project, + commands::project::delete_project, + commands::credential::set_tuleap_credentials, + commands::credential::get_tuleap_credentials, + commands::credential::delete_tuleap_credentials, + commands::credential::test_tuleap_connection, + commands::tracker::add_tracker, + commands::tracker::list_trackers, + commands::tracker::update_tracker, + commands::tracker::remove_tracker, + commands::tracker::get_tracker_fields, + commands::tracker::list_processed_tickets, + commands::poller::manual_poll, + commands::poller::get_queue_status, + commands::orchestrator::get_ticket_result, + commands::orchestrator::retry_ticket, + commands::orchestrator::cancel_ticket, + commands::worktree::list_worktrees, + commands::worktree::get_worktree_diff, + commands::worktree::apply_fix_to_branch, + commands::worktree::delete_worktree_cmd, + commands::worktree::list_local_branches, + ]) +``` + +- [ ] **Step 2: Verify full build compiles** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo check 2>&1 | tail -10` +Expected: compiles without errors + +- [ ] **Step 3: Run all tests** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo test 2>&1 | tail -20` +Expected: all tests pass (existing + new) + +- [ ] **Step 4: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src-tauri/src/lib.rs +git commit -m "feat: wire orchestrator startup and register Phase 3 Tauri commands" +``` + +--- + +### Task 7: Frontend dependencies, types, and API wrappers + +**Files:** +- Modify: `src/lib/types.ts` +- Modify: `src/lib/api.ts` +- Modify: `package.json` (via npm install) + +- [ ] **Step 1: Install frontend dependencies** + +```bash +cd /home/leclere/Projets/orchai && npm install react-markdown remark-gfm +``` + +- [ ] **Step 2: Add Worktree and TicketResult types to types.ts** + +Append to `src/lib/types.ts`: + +```typescript +export interface Worktree { + id: string; + ticket_id: string; + path: string; + branch_name: string; + status: string; + created_at: string; + merged_at: string | null; + merged_into: string | null; +} + +export interface TicketResult { + ticket: ProcessedTicket; + worktree: Worktree | null; +} +``` + +- [ ] **Step 3: Add API wrappers to api.ts** + +Add the import for the new types in `src/lib/api.ts`: + +```typescript +import type { + Project, + TuleapCredentialsSafe, + AgentConfig, + FilterGroup, + WatchedTracker, + TrackerField, + ProcessedTicket, + Worktree, + TicketResult, +} from "./types"; +``` + +Append these functions to `src/lib/api.ts`: + +```typescript +// Orchestrator +export async function getTicketResult(ticketId: string): Promise { + return invoke("get_ticket_result", { ticketId }); +} +export async function retryTicket(ticketId: string): Promise { + return invoke("retry_ticket", { ticketId }); +} +export async function cancelTicket(ticketId: string): Promise { + return invoke("cancel_ticket", { ticketId }); +} + +// Worktrees +export async function listWorktrees(projectId: string): Promise { + return invoke("list_worktrees", { projectId }); +} +export async function getWorktreeDiff(worktreeId: string): Promise { + return invoke("get_worktree_diff", { worktreeId }); +} +export async function applyFixToBranch(worktreeId: string, targetBranch: string): Promise { + return invoke("apply_fix_to_branch", { worktreeId, targetBranch }); +} +export async function deleteWorktreeCmd(worktreeId: string): Promise { + return invoke("delete_worktree_cmd", { worktreeId }); +} +export async function listLocalBranches(projectId: string): Promise { + return invoke("list_local_branches", { projectId }); +} +``` + +- [ ] **Step 4: Verify TypeScript compiles** + +Run: `cd /home/leclere/Projets/orchai && npx tsc --noEmit 2>&1 | tail -10` +Expected: no errors + +- [ ] **Step 5: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add package.json package-lock.json src/lib/types.ts src/lib/api.ts +git commit -m "feat: add frontend types, API wrappers, and markdown deps for Phase 3" +``` + +--- + +### Task 8: Ticket list page + +**Files:** +- Create: `src/components/tickets/TicketList.tsx` +- Modify: `src/components/projects/ProjectDashboard.tsx` +- Modify: `src/App.tsx` + +- [ ] **Step 1: Create TicketList component** + +Create `src/components/tickets/TicketList.tsx`: + +```tsx +import { useEffect, useState } from "react"; +import { useParams, Link } from "react-router-dom"; +import { listProcessedTickets, getProject } from "../../lib/api"; +import type { ProcessedTicket, Project } from "../../lib/types"; + +function statusBadgeClass(status: string): string { + switch (status) { + case "Pending": + return "bg-yellow-100 text-yellow-700"; + case "Analyzing": + return "bg-blue-100 text-blue-700"; + case "Developing": + return "bg-purple-100 text-purple-700"; + case "Done": + return "bg-green-100 text-green-700"; + case "Error": + return "bg-red-100 text-red-700"; + case "Cancelled": + return "bg-gray-100 text-gray-500"; + default: + return "bg-gray-100 text-gray-700"; + } +} + +export default function TicketList() { + const { projectId } = useParams(); + const [project, setProject] = useState(null); + const [tickets, setTickets] = useState([]); + const [filter, setFilter] = useState("all"); + + useEffect(() => { + if (!projectId) return; + Promise.all([getProject(projectId), listProcessedTickets(projectId)]).then( + ([proj, tkts]) => { + setProject(proj); + setTickets(tkts); + } + ); + }, [projectId]); + + const filtered = + filter === "all" ? tickets : tickets.filter((t) => t.status === filter); + + return ( +
+
+
+ + {project?.name} + +

Processed Tickets

+
+
+ +
+ {["all", "Pending", "Analyzing", "Developing", "Done", "Error"].map( + (s) => ( + + ) + )} +
+ + {filtered.length === 0 ? ( +
+ No tickets found. +
+ ) : ( +
+ {filtered.map((ticket) => ( + +
+
+
+ + #{ticket.artifact_id} + + + {ticket.artifact_title} + +
+
+ {new Date(ticket.detected_at).toLocaleString()} + {ticket.processed_at && ( + + Processed: {new Date(ticket.processed_at).toLocaleString()} + + )} +
+
+ + {ticket.status} + +
+ + ))} +
+ )} +
+ ); +} +``` + +- [ ] **Step 2: Make dashboard ticket items link to detail view** + +In `src/components/projects/ProjectDashboard.tsx`, wrap each ticket in the "Recent Tickets" section with a `Link`. Replace the ticket `
` with: + +Change the ticket rendering from a plain `
` to a ``: + +```tsx + +``` + +Close with `` instead of `
`. + +Also add a "View all tickets" link after the recent tickets list, and a `Link` to the ticket list page in the section header: + +Replace the "Recent Tickets" header with: + +```tsx +
+

Recent Tickets

+ {tickets.length > 0 && ( + + View all ({tickets.length}) + + )} +
+``` + +- [ ] **Step 3: Add routes to App.tsx** + +In `src/App.tsx`, add imports: + +```tsx +import TicketList from "./components/tickets/TicketList"; +``` + +Add routes inside the `}>` block: + +```tsx + } /> +``` + +- [ ] **Step 4: Verify TypeScript compiles** + +Run: `cd /home/leclere/Projets/orchai && npx tsc --noEmit 2>&1 | tail -10` +Expected: no errors + +- [ ] **Step 5: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src/components/tickets/TicketList.tsx src/components/projects/ProjectDashboard.tsx src/App.tsx +git commit -m "feat: add ticket list page with status filtering and dashboard links" +``` + +--- + +### Task 9: Ticket detail page with markdown reports, diff viewer, and actions + +**Files:** +- Create: `src/components/tickets/TicketDetail.tsx` +- Modify: `src/App.tsx` + +- [ ] **Step 1: Create TicketDetail component** + +Create `src/components/tickets/TicketDetail.tsx`: + +```tsx +import { useEffect, useState } from "react"; +import { useParams, useNavigate } from "react-router-dom"; +import Markdown from "react-markdown"; +import remarkGfm from "remark-gfm"; +import { + getTicketResult, + retryTicket, + cancelTicket, + getWorktreeDiff, + applyFixToBranch, + deleteWorktreeCmd, + listLocalBranches, +} from "../../lib/api"; +import type { ProcessedTicket, Worktree } from "../../lib/types"; + +function statusBadgeClass(status: string): string { + switch (status) { + case "Pending": + return "bg-yellow-100 text-yellow-700"; + case "Analyzing": + return "bg-blue-100 text-blue-700"; + case "Developing": + return "bg-purple-100 text-purple-700"; + case "Done": + return "bg-green-100 text-green-700"; + case "Error": + return "bg-red-100 text-red-700"; + case "Cancelled": + return "bg-gray-100 text-gray-500"; + default: + return "bg-gray-100 text-gray-700"; + } +} + +function DiffViewer({ diff }: { diff: string }) { + if (!diff) { + return ( +
+ No changes detected. +
+ ); + } + + const lines = diff.split("\n"); + return ( +
+      {lines.map((line, i) => {
+        let cls = "";
+        if (line.startsWith("+++") || line.startsWith("---"))
+          cls = "text-gray-400";
+        else if (line.startsWith("+")) cls = "text-green-400 bg-green-900/20";
+        else if (line.startsWith("-")) cls = "text-red-400 bg-red-900/20";
+        else if (line.startsWith("@@")) cls = "text-blue-400";
+        else if (line.startsWith("diff ")) cls = "text-yellow-400 font-bold";
+        return (
+          
+ {line} +
+ ); + })} +
+ ); +} + +export default function TicketDetail() { + const { ticketId } = useParams(); + const navigate = useNavigate(); + const [ticket, setTicket] = useState(null); + const [worktree, setWorktree] = useState(null); + const [diff, setDiff] = useState(null); + const [branches, setBranches] = useState([]); + const [targetBranch, setTargetBranch] = useState(""); + const [tab, setTab] = useState<"info" | "analyst" | "developer" | "diff">( + "info" + ); + const [loading, setLoading] = useState(false); + const [error, setError] = useState(""); + + async function loadData() { + if (!ticketId) return; + try { + const result = await getTicketResult(ticketId); + setTicket(result.ticket); + setWorktree(result.worktree); + + // Auto-select the most relevant tab + if (result.ticket.developer_report) setTab("developer"); + else if (result.ticket.analyst_report) setTab("analyst"); + + // Load diff if worktree exists + if (result.worktree && result.worktree.status === "Active") { + try { + const d = await getWorktreeDiff(result.worktree.id); + setDiff(d); + } catch { + setDiff(null); + } + } + } catch (err) { + setError(err instanceof Error ? err.message : String(err)); + } + } + + useEffect(() => { + loadData(); + }, [ticketId]); + + async function handleRetry() { + if (!ticketId) return; + setLoading(true); + try { + await retryTicket(ticketId); + await loadData(); + } catch (err) { + setError(err instanceof Error ? err.message : String(err)); + } + setLoading(false); + } + + async function handleCancel() { + if (!ticketId) return; + setLoading(true); + try { + await cancelTicket(ticketId); + await loadData(); + } catch (err) { + setError(err instanceof Error ? err.message : String(err)); + } + setLoading(false); + } + + async function handleApplyFix() { + if (!worktree || !targetBranch) return; + setLoading(true); + setError(""); + try { + await applyFixToBranch(worktree.id, targetBranch); + await loadData(); + } catch (err) { + setError(err instanceof Error ? err.message : String(err)); + } + setLoading(false); + } + + async function handleDeleteWorktree() { + if (!worktree) return; + if (!window.confirm("Delete this worktree and its branch?")) return; + setLoading(true); + try { + await deleteWorktreeCmd(worktree.id); + setWorktree(null); + setDiff(null); + } catch (err) { + setError(err instanceof Error ? err.message : String(err)); + } + setLoading(false); + } + + async function loadBranches() { + if (!ticket) return; + try { + // We need the project ID -- navigate through the ticket data + // The tracker_id on the ticket links to the watched tracker + // For simplicity, parse project_id from the URL if available, or use a dedicated call + // Since we already have the worktree, we can get branches through the ticket result + const result = await getTicketResult(ticket.id); + // Get project_id through the ticket -> tracker chain is done server-side + // For branches, we need the project_id. Let's add it to the load flow. + // Actually, we'll use a simple approach: the branches list is loaded on demand + // when the user opens the "Apply fix" section + } catch { + // ignore + } + } + + if (!ticket) { + return
Loading...
; + } + + const tabs = [ + { key: "info" as const, label: "Info" }, + { + key: "analyst" as const, + label: "Analyst Report", + disabled: !ticket.analyst_report, + }, + { + key: "developer" as const, + label: "Developer Report", + disabled: !ticket.developer_report, + }, + { key: "diff" as const, label: "Diff", disabled: !diff && !worktree }, + ]; + + return ( +
+ {/* Header */} +
+
+ +

+ + #{ticket.artifact_id} + + {ticket.artifact_title} + + {ticket.status} + +

+
+
+ {(ticket.status === "Error" || + ticket.status === "Done" || + ticket.status === "Cancelled") && ( + + )} + {(ticket.status === "Pending" || + ticket.status === "Analyzing" || + ticket.status === "Developing") && ( + + )} +
+
+ + {error && ( +
+ {error} +
+ )} + + {/* Tabs */} +
+ {tabs.map((t) => ( + + ))} +
+ + {/* Tab content */} + {tab === "info" && ( +
+
+
+ Status: + {ticket.status} +
+
+ Detected: + + {new Date(ticket.detected_at).toLocaleString()} + +
+ {ticket.processed_at && ( +
+ Processed: + + {new Date(ticket.processed_at).toLocaleString()} + +
+ )} + {worktree && ( +
+ Worktree: + + {worktree.branch_name} + + + {worktree.status} + +
+ )} +
+ + {/* Worktree actions */} + {worktree && worktree.status === "Active" && ( +
+

Worktree Actions

+
+ setTargetBranch(e.target.value)} + className="flex-1 px-3 py-1.5 border border-gray-300 rounded text-sm focus:ring-2 focus:ring-blue-500 focus:border-transparent" + /> + +
+ +
+ )} + + {worktree && worktree.status === "Merged" && ( +
+ Fix applied to branch: {worktree.merged_into} +
+ )} +
+ )} + + {tab === "analyst" && ticket.analyst_report && ( +
+ + {ticket.analyst_report} + +
+ )} + + {tab === "developer" && ticket.developer_report && ( +
+ + {ticket.developer_report} + +
+ )} + + {tab === "diff" && } +
+ ); +} +``` + +- [ ] **Step 2: Add ticket detail route to App.tsx** + +In `src/App.tsx`, add the import: + +```tsx +import TicketDetail from "./components/tickets/TicketDetail"; +``` + +Add the route inside `}>`: + +```tsx + } /> +``` + +- [ ] **Step 3: Verify TypeScript compiles** + +Run: `cd /home/leclere/Projets/orchai && npx tsc --noEmit 2>&1 | tail -10` +Expected: no errors + +- [ ] **Step 4: Commit** + +```bash +cd /home/leclere/Projets/orchai +git add src/components/tickets/TicketDetail.tsx src/App.tsx +git commit -m "feat: add ticket detail page with markdown reports, diff viewer, and worktree actions" +``` + +--- + +### Task 10: Final verification + +- [ ] **Step 1: Run all backend tests** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo test 2>&1 | tail -30` +Expected: all tests pass, including new tests from Tasks 1-4 + +- [ ] **Step 2: Run clippy** + +Run: `cd /home/leclere/Projets/orchai/src-tauri && cargo clippy -- -D warnings 2>&1 | tail -20` +Expected: no warnings + +- [ ] **Step 3: Verify frontend builds** + +Run: `cd /home/leclere/Projets/orchai && npx tsc --noEmit && echo "OK"` +Expected: "OK" + +- [ ] **Step 4: Commit any fixes needed from verification** + +If clippy or tsc produced warnings/errors, fix them and commit: + +```bash +cd /home/leclere/Projets/orchai +git add -u +git commit -m "fix: resolve clippy warnings and TypeScript errors from Phase 3" +``` diff --git a/index.html b/index.html new file mode 100644 index 0000000..bda81b6 --- /dev/null +++ b/index.html @@ -0,0 +1,12 @@ + + + + + + Orchai + + +
+ + + diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..7d5ea9a --- /dev/null +++ b/package-lock.json @@ -0,0 +1,2757 @@ +{ + "name": "orchai", + "version": "0.1.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "orchai", + "version": "0.1.0", + "dependencies": { + "@tauri-apps/api": "^2.10.1", + "@tauri-apps/plugin-dialog": "^2.7.0", + "react": "^18.3.1", + "react-dom": "^18.3.1", + "react-router-dom": "^7.14.0" + }, + "devDependencies": { + "@tailwindcss/vite": "^4.2.2", + "@tauri-apps/cli": "^2", + "@types/react": "^18.3.1", + "@types/react-dom": "^18.3.1", + "@vitejs/plugin-react": "^4.3.1", + "tailwindcss": "^4.2.2", + "typescript": "^5.6.3", + "vite": "^6.0.3" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz", + "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.28.5", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.0.tgz", + "integrity": "sha512-T1NCJqT/j9+cn8fvkt7jtwbLBfLC/1y1c7NtCeXFRgzGTsafi68MRv8yzkYSapBnFA6L3U2VSc02ciDzoAJhJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz", + "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-module-transforms": "^7.28.6", + "@babel/helpers": "^7.28.6", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/traverse": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/generator": { + "version": "7.29.1", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz", + "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz", + "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.28.6", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz", + "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz", + "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.28.6", + "@babel/helper-validator-identifier": "^7.28.5", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.28.6.tgz", + "integrity": "sha512-S9gzZ/bz83GRysI7gAD4wPT/AI3uCnY+9xn+Mx/KPs2JwHJIz1W8PZkg2cqyt3RNOBM8ejcXhV6y8Og7ly/Dug==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", + "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.29.2", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.29.2.tgz", + "integrity": "sha512-HoGuUs4sCZNezVEKdVcwqmZN8GoHirLUcLaYVNBK2J0DadGtdcqgr3BCbvH8+XUo4NGjNl3VOtSjEKNzqfFgKw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.29.2", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.2.tgz", + "integrity": "sha512-4GgRzy/+fsBa72/RZVJmGKPmZu9Byn8o4MoLpmNe1m8ZfYnz5emHLQz3U4gLud6Zwl0RZIcgiLD7Uq7ySFuDLA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.29.0" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-self": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.27.1.tgz", + "integrity": "sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-source": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.27.1.tgz", + "integrity": "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/template": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz", + "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.28.6", + "@babel/parser": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz", + "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz", + "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz", + "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz", + "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz", + "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz", + "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz", + "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz", + "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz", + "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz", + "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz", + "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz", + "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz", + "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz", + "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz", + "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz", + "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz", + "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz", + "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz", + "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz", + "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz", + "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz", + "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz", + "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openharmony-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz", + "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz", + "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz", + "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz", + "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz", + "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", + "dev": true, + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@rolldown/pluginutils": { + "version": "1.0.0-beta.27", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.27.tgz", + "integrity": "sha512-+d0F4MKMCbeVUJwG96uQ4SgAznZNSq93I3V+9NHA4OpvqG8mRCpGdKmK8l/dl02h2CCDHwW2FqilnTyDcAnqjA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.60.1.tgz", + "integrity": "sha512-d6FinEBLdIiK+1uACUttJKfgZREXrF0Qc2SmLII7W2AD8FfiZ9Wjd+rD/iRuf5s5dWrr1GgwXCvPqOuDquOowA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.60.1.tgz", + "integrity": "sha512-YjG/EwIDvvYI1YvYbHvDz/BYHtkY4ygUIXHnTdLhG+hKIQFBiosfWiACWortsKPKU/+dUwQQCKQM3qrDe8c9BA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.60.1.tgz", + "integrity": "sha512-mjCpF7GmkRtSJwon+Rq1N8+pI+8l7w5g9Z3vWj4T7abguC4Czwi3Yu/pFaLvA3TTeMVjnu3ctigusqWUfjZzvw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.60.1.tgz", + "integrity": "sha512-haZ7hJ1JT4e9hqkoT9R/19XW2QKqjfJVv+i5AGg57S+nLk9lQnJ1F/eZloRO3o9Scy9CM3wQ9l+dkXtcBgN5Ew==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.60.1.tgz", + "integrity": "sha512-czw90wpQq3ZsAVBlinZjAYTKduOjTywlG7fEeWKUA7oCmpA8xdTkxZZlwNJKWqILlq0wehoZcJYfBvOyhPTQ6w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.60.1.tgz", + "integrity": "sha512-KVB2rqsxTHuBtfOeySEyzEOB7ltlB/ux38iu2rBQzkjbwRVlkhAGIEDiiYnO2kFOkJp+Z7pUXKyrRRFuFUKt+g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.60.1.tgz", + "integrity": "sha512-L+34Qqil+v5uC0zEubW7uByo78WOCIrBvci69E7sFASRl0X7b/MB6Cqd1lky/CtcSVTydWa2WZwFuWexjS5o6g==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.60.1.tgz", + "integrity": "sha512-n83O8rt4v34hgFzlkb1ycniJh7IR5RCIqt6mz1VRJD6pmhRi0CXdmfnLu9dIUS6buzh60IvACM842Ffb3xd6Gg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.60.1.tgz", + "integrity": "sha512-Nql7sTeAzhTAja3QXeAI48+/+GjBJ+QmAH13snn0AJSNL50JsDqotyudHyMbO2RbJkskbMbFJfIJKWA6R1LCJQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.60.1.tgz", + "integrity": "sha512-+pUymDhd0ys9GcKZPPWlFiZ67sTWV5UU6zOJat02M1+PiuSGDziyRuI/pPue3hoUwm2uGfxdL+trT6Z9rxnlMA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.60.1.tgz", + "integrity": "sha512-VSvgvQeIcsEvY4bKDHEDWcpW4Yw7BtlKG1GUT4FzBUlEKQK0rWHYBqQt6Fm2taXS+1bXvJT6kICu5ZwqKCnvlQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.60.1.tgz", + "integrity": "sha512-4LqhUomJqwe641gsPp6xLfhqWMbQV04KtPp7/dIp0nzPxAkNY1AbwL5W0MQpcalLYk07vaW9Kp1PBhdpZYYcEw==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.60.1.tgz", + "integrity": "sha512-tLQQ9aPvkBxOc/EUT6j3pyeMD6Hb8QF2BTBnCQWP/uu1lhc9AIrIjKnLYMEroIz/JvtGYgI9dF3AxHZNaEH0rw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.60.1.tgz", + "integrity": "sha512-RMxFhJwc9fSXP6PqmAz4cbv3kAyvD1etJFjTx4ONqFP9DkTkXsAMU4v3Vyc5BgzC+anz7nS/9tp4obsKfqkDHg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.60.1.tgz", + "integrity": "sha512-QKgFl+Yc1eEk6MmOBfRHYF6lTxiiiV3/z/BRrbSiW2I7AFTXoBFvdMEyglohPj//2mZS4hDOqeB0H1ACh3sBbg==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.60.1.tgz", + "integrity": "sha512-RAjXjP/8c6ZtzatZcA1RaQr6O1TRhzC+adn8YZDnChliZHviqIjmvFwHcxi4JKPSDAt6Uhf/7vqcBzQJy0PDJg==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.60.1.tgz", + "integrity": "sha512-wcuocpaOlaL1COBYiA89O6yfjlp3RwKDeTIA0hM7OpmhR1Bjo9j31G1uQVpDlTvwxGn2nQs65fBFL5UFd76FcQ==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.60.1.tgz", + "integrity": "sha512-77PpsFQUCOiZR9+LQEFg9GClyfkNXj1MP6wRnzYs0EeWbPcHs02AXu4xuUbM1zhwn3wqaizle3AEYg5aeoohhg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.60.1.tgz", + "integrity": "sha512-5cIATbk5vynAjqqmyBjlciMJl1+R/CwX9oLk/EyiFXDWd95KpHdrOJT//rnUl4cUcskrd0jCCw3wpZnhIHdD9w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-openbsd-x64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.60.1.tgz", + "integrity": "sha512-cl0w09WsCi17mcmWqqglez9Gk8isgeWvoUZ3WiJFYSR3zjBQc2J5/ihSjpl+VLjPqjQ/1hJRcqBfLjssREQILw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ] + }, + "node_modules/@rollup/rollup-openharmony-arm64": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.60.1.tgz", + "integrity": "sha512-4Cv23ZrONRbNtbZa37mLSueXUCtN7MXccChtKpUnQNgF010rjrjfHx3QxkS2PI7LqGT5xXyYs1a7LbzAwT0iCA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.60.1.tgz", + "integrity": "sha512-i1okWYkA4FJICtr7KpYzFpRTHgy5jdDbZiWfvny21iIKky5YExiDXP+zbXzm3dUcFpkEeYNHgQ5fuG236JPq0g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.60.1.tgz", + "integrity": "sha512-u09m3CuwLzShA0EYKMNiFgcjjzwqtUMLmuCJLeZWjjOYA3IT2Di09KaxGBTP9xVztWyIWjVdsB2E9goMjZvTQg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-gnu": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.60.1.tgz", + "integrity": "sha512-k+600V9Zl1CM7eZxJgMyTUzmrmhB/0XZnF4pRypKAlAgxmedUA+1v9R+XOFv56W4SlHEzfeMtzujLJD22Uz5zg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.60.1.tgz", + "integrity": "sha512-lWMnixq/QzxyhTV6NjQJ4SFo1J6PvOX8vUx5Wb4bBPsEb+8xZ89Bz6kOXpfXj9ak9AHTQVQzlgzBEc1SyM27xQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@tailwindcss/node": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/node/-/node-4.2.2.tgz", + "integrity": "sha512-pXS+wJ2gZpVXqFaUEjojq7jzMpTGf8rU6ipJz5ovJV6PUGmlJ+jvIwGrzdHdQ80Sg+wmQxUFuoW1UAAwHNEdFA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/remapping": "^2.3.5", + "enhanced-resolve": "^5.19.0", + "jiti": "^2.6.1", + "lightningcss": "1.32.0", + "magic-string": "^0.30.21", + "source-map-js": "^1.2.1", + "tailwindcss": "4.2.2" + } + }, + "node_modules/@tailwindcss/oxide": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide/-/oxide-4.2.2.tgz", + "integrity": "sha512-qEUA07+E5kehxYp9BVMpq9E8vnJuBHfJEC0vPC5e7iL/hw7HR61aDKoVoKzrG+QKp56vhNZe4qwkRmMC0zDLvg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 20" + }, + "optionalDependencies": { + "@tailwindcss/oxide-android-arm64": "4.2.2", + "@tailwindcss/oxide-darwin-arm64": "4.2.2", + "@tailwindcss/oxide-darwin-x64": "4.2.2", + "@tailwindcss/oxide-freebsd-x64": "4.2.2", + "@tailwindcss/oxide-linux-arm-gnueabihf": "4.2.2", + "@tailwindcss/oxide-linux-arm64-gnu": "4.2.2", + "@tailwindcss/oxide-linux-arm64-musl": "4.2.2", + "@tailwindcss/oxide-linux-x64-gnu": "4.2.2", + "@tailwindcss/oxide-linux-x64-musl": "4.2.2", + "@tailwindcss/oxide-wasm32-wasi": "4.2.2", + "@tailwindcss/oxide-win32-arm64-msvc": "4.2.2", + "@tailwindcss/oxide-win32-x64-msvc": "4.2.2" + } + }, + "node_modules/@tailwindcss/oxide-android-arm64": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-android-arm64/-/oxide-android-arm64-4.2.2.tgz", + "integrity": "sha512-dXGR1n+P3B6748jZO/SvHZq7qBOqqzQ+yFrXpoOWWALWndF9MoSKAT3Q0fYgAzYzGhxNYOoysRvYlpixRBBoDg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-darwin-arm64": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-arm64/-/oxide-darwin-arm64-4.2.2.tgz", + "integrity": "sha512-iq9Qjr6knfMpZHj55/37ouZeykwbDqF21gPFtfnhCCKGDcPI/21FKC9XdMO/XyBM7qKORx6UIhGgg6jLl7BZlg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-darwin-x64": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-x64/-/oxide-darwin-x64-4.2.2.tgz", + "integrity": "sha512-BlR+2c3nzc8f2G639LpL89YY4bdcIdUmiOOkv2GQv4/4M0vJlpXEa0JXNHhCHU7VWOKWT/CjqHdTP8aUuDJkuw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-freebsd-x64": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-freebsd-x64/-/oxide-freebsd-x64-4.2.2.tgz", + "integrity": "sha512-YUqUgrGMSu2CDO82hzlQ5qSb5xmx3RUrke/QgnoEx7KvmRJHQuZHZmZTLSuuHwFf0DJPybFMXMYf+WJdxHy/nQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-linux-arm-gnueabihf": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm-gnueabihf/-/oxide-linux-arm-gnueabihf-4.2.2.tgz", + "integrity": "sha512-FPdhvsW6g06T9BWT0qTwiVZYE2WIFo2dY5aCSpjG/S/u1tby+wXoslXS0kl3/KXnULlLr1E3NPRRw0g7t2kgaQ==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-linux-arm64-gnu": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-gnu/-/oxide-linux-arm64-gnu-4.2.2.tgz", + "integrity": "sha512-4og1V+ftEPXGttOO7eCmW7VICmzzJWgMx+QXAJRAhjrSjumCwWqMfkDrNu1LXEQzNAwz28NCUpucgQPrR4S2yw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-linux-arm64-musl": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-musl/-/oxide-linux-arm64-musl-4.2.2.tgz", + "integrity": "sha512-oCfG/mS+/+XRlwNjnsNLVwnMWYH7tn/kYPsNPh+JSOMlnt93mYNCKHYzylRhI51X+TbR+ufNhhKKzm6QkqX8ag==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-linux-x64-gnu": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-gnu/-/oxide-linux-x64-gnu-4.2.2.tgz", + "integrity": "sha512-rTAGAkDgqbXHNp/xW0iugLVmX62wOp2PoE39BTCGKjv3Iocf6AFbRP/wZT/kuCxC9QBh9Pu8XPkv/zCZB2mcMg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-linux-x64-musl": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-musl/-/oxide-linux-x64-musl-4.2.2.tgz", + "integrity": "sha512-XW3t3qwbIwiSyRCggeO2zxe3KWaEbM0/kW9e8+0XpBgyKU4ATYzcVSMKteZJ1iukJ3HgHBjbg9P5YPRCVUxlnQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-wasm32-wasi": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-wasm32-wasi/-/oxide-wasm32-wasi-4.2.2.tgz", + "integrity": "sha512-eKSztKsmEsn1O5lJ4ZAfyn41NfG7vzCg496YiGtMDV86jz1q/irhms5O0VrY6ZwTUkFy/EKG3RfWgxSI3VbZ8Q==", + "bundleDependencies": [ + "@napi-rs/wasm-runtime", + "@emnapi/core", + "@emnapi/runtime", + "@tybys/wasm-util", + "@emnapi/wasi-threads", + "tslib" + ], + "cpu": [ + "wasm32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/core": "^1.8.1", + "@emnapi/runtime": "^1.8.1", + "@emnapi/wasi-threads": "^1.1.0", + "@napi-rs/wasm-runtime": "^1.1.1", + "@tybys/wasm-util": "^0.10.1", + "tslib": "^2.8.1" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@tailwindcss/oxide-win32-arm64-msvc": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-arm64-msvc/-/oxide-win32-arm64-msvc-4.2.2.tgz", + "integrity": "sha512-qPmaQM4iKu5mxpsrWZMOZRgZv1tOZpUm+zdhhQP0VhJfyGGO3aUKdbh3gDZc/dPLQwW4eSqWGrrcWNBZWUWaXQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/oxide-win32-x64-msvc": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-x64-msvc/-/oxide-win32-x64-msvc-4.2.2.tgz", + "integrity": "sha512-1T/37VvI7WyH66b+vqHj/cLwnCxt7Qt3WFu5Q8hk65aOvlwAhs7rAp1VkulBJw/N4tMirXjVnylTR72uI0HGcA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 20" + } + }, + "node_modules/@tailwindcss/vite": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@tailwindcss/vite/-/vite-4.2.2.tgz", + "integrity": "sha512-mEiF5HO1QqCLXoNEfXVA1Tzo+cYsrqV7w9Juj2wdUFyW07JRenqMG225MvPwr3ZD9N1bFQj46X7r33iHxLUW0w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@tailwindcss/node": "4.2.2", + "@tailwindcss/oxide": "4.2.2", + "tailwindcss": "4.2.2" + }, + "peerDependencies": { + "vite": "^5.2.0 || ^6 || ^7 || ^8" + } + }, + "node_modules/@tauri-apps/api": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/api/-/api-2.10.1.tgz", + "integrity": "sha512-hKL/jWf293UDSUN09rR69hrToyIXBb8CjGaWC7gfinvnQrBVvnLr08FeFi38gxtugAVyVcTa5/FD/Xnkb1siBw==", + "license": "Apache-2.0 OR MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/tauri" + } + }, + "node_modules/@tauri-apps/cli": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli/-/cli-2.10.1.tgz", + "integrity": "sha512-jQNGF/5quwORdZSSLtTluyKQ+o6SMa/AUICfhf4egCGFdMHqWssApVgYSbg+jmrZoc8e1DscNvjTnXtlHLS11g==", + "dev": true, + "license": "Apache-2.0 OR MIT", + "bin": { + "tauri": "tauri.js" + }, + "engines": { + "node": ">= 10" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/tauri" + }, + "optionalDependencies": { + "@tauri-apps/cli-darwin-arm64": "2.10.1", + "@tauri-apps/cli-darwin-x64": "2.10.1", + "@tauri-apps/cli-linux-arm-gnueabihf": "2.10.1", + "@tauri-apps/cli-linux-arm64-gnu": "2.10.1", + "@tauri-apps/cli-linux-arm64-musl": "2.10.1", + "@tauri-apps/cli-linux-riscv64-gnu": "2.10.1", + "@tauri-apps/cli-linux-x64-gnu": "2.10.1", + "@tauri-apps/cli-linux-x64-musl": "2.10.1", + "@tauri-apps/cli-win32-arm64-msvc": "2.10.1", + "@tauri-apps/cli-win32-ia32-msvc": "2.10.1", + "@tauri-apps/cli-win32-x64-msvc": "2.10.1" + } + }, + "node_modules/@tauri-apps/cli-darwin-arm64": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-darwin-arm64/-/cli-darwin-arm64-2.10.1.tgz", + "integrity": "sha512-Z2OjCXiZ+fbYZy7PmP3WRnOpM9+Fy+oonKDEmUE6MwN4IGaYqgceTjwHucc/kEEYZos5GICve35f7ZiizgqEnQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-darwin-x64": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-darwin-x64/-/cli-darwin-x64-2.10.1.tgz", + "integrity": "sha512-V/irQVvjPMGOTQqNj55PnQPVuH4VJP8vZCN7ajnj+ZS8Kom1tEM2hR3qbbIRoS3dBKs5mbG8yg1WC+97dq17Pw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-arm-gnueabihf": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-arm-gnueabihf/-/cli-linux-arm-gnueabihf-2.10.1.tgz", + "integrity": "sha512-Hyzwsb4VnCWKGfTw+wSt15Z2pLw2f0JdFBfq2vHBOBhvg7oi6uhKiF87hmbXOBXUZaGkyRDkCHsdzJcIfoJC2w==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-arm64-gnu": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-arm64-gnu/-/cli-linux-arm64-gnu-2.10.1.tgz", + "integrity": "sha512-OyOYs2t5GkBIvyWjA1+h4CZxTcdz1OZPCWAPz5DYEfB0cnWHERTnQ/SLayQzncrT0kwRoSfSz9KxenkyJoTelA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-arm64-musl": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-arm64-musl/-/cli-linux-arm64-musl-2.10.1.tgz", + "integrity": "sha512-MIj78PDDGjkg3NqGptDOGgfXks7SYJwhiMh8SBoZS+vfdz7yP5jN18bNaLnDhsVIPARcAhE1TlsZe/8Yxo2zqg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-riscv64-gnu": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-riscv64-gnu/-/cli-linux-riscv64-gnu-2.10.1.tgz", + "integrity": "sha512-X0lvOVUg8PCVaoEtEAnpxmnkwlE1gcMDTqfhbefICKDnOTJ5Est3qL0SrWxizDackIOKBcvtpejrSiVpuJI1kw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-x64-gnu": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-x64-gnu/-/cli-linux-x64-gnu-2.10.1.tgz", + "integrity": "sha512-2/12bEzsJS9fAKybxgicCDFxYD1WEI9kO+tlDwX5znWG2GwMBaiWcmhGlZ8fi+DMe9CXlcVarMTYc0L3REIRxw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-x64-musl": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-x64-musl/-/cli-linux-x64-musl-2.10.1.tgz", + "integrity": "sha512-Y8J0ZzswPz50UcGOFuXGEMrxbjwKSPgXftx5qnkuMs2rmwQB5ssvLb6tn54wDSYxe7S6vlLob9vt0VKuNOaCIQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-win32-arm64-msvc": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-win32-arm64-msvc/-/cli-win32-arm64-msvc-2.10.1.tgz", + "integrity": "sha512-iSt5B86jHYAPJa/IlYw++SXtFPGnWtFJriHn7X0NFBVunF6zu9+/zOn8OgqIWSl8RgzhLGXQEEtGBdR4wzpVgg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-win32-ia32-msvc": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-win32-ia32-msvc/-/cli-win32-ia32-msvc-2.10.1.tgz", + "integrity": "sha512-gXyxgEzsFegmnWywYU5pEBURkcFN/Oo45EAwvZrHMh+zUSEAvO5E8TXsgPADYm31d1u7OQU3O3HsYfVBf2moHw==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-win32-x64-msvc": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-win32-x64-msvc/-/cli-win32-x64-msvc-2.10.1.tgz", + "integrity": "sha512-6Cn7YpPFwzChy0ERz6djKEmUehWrYlM+xTaNzGPgZocw3BD7OfwfWHKVWxXzdjEW2KfKkHddfdxK1XXTYqBRLg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/plugin-dialog": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/@tauri-apps/plugin-dialog/-/plugin-dialog-2.7.0.tgz", + "integrity": "sha512-4nS/hfGMGCXiAS3LtVjH9AgsSAPJeG/7R+q8agTFqytjnMa4Zq95Bq8WzVDkckpanX+yyRHXnRtrKXkANKDHvw==", + "license": "MIT OR Apache-2.0", + "dependencies": { + "@tauri-apps/api": "^2.10.1" + } + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz", + "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", + "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.2" + } + }, + "node_modules/@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/prop-types": { + "version": "15.7.15", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz", + "integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/react": { + "version": "18.3.28", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.28.tgz", + "integrity": "sha512-z9VXpC7MWrhfWipitjNdgCauoMLRdIILQsAEV+ZesIzBq/oUlxk0m3ApZuMFCXdnS4U7KrI+l3WRUEGQ8K1QKw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@types/prop-types": "*", + "csstype": "^3.2.2" + } + }, + "node_modules/@types/react-dom": { + "version": "18.3.7", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.7.tgz", + "integrity": "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "@types/react": "^18.0.0" + } + }, + "node_modules/@vitejs/plugin-react": { + "version": "4.7.0", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.7.0.tgz", + "integrity": "sha512-gUu9hwfWvvEDBBmgtAowQCojwZmJ5mcLn3aufeCsitijs3+f2NsrPtlAWIR6OPiqljl96GVCUbLe0HyqIpVaoA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.28.0", + "@babel/plugin-transform-react-jsx-self": "^7.27.1", + "@babel/plugin-transform-react-jsx-source": "^7.27.1", + "@rolldown/pluginutils": "1.0.0-beta.27", + "@types/babel__core": "^7.20.5", + "react-refresh": "^0.17.0" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "peerDependencies": { + "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0" + } + }, + "node_modules/baseline-browser-mapping": { + "version": "2.10.18", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.18.tgz", + "integrity": "sha512-VSnGQAOLtP5mib/DPyg2/t+Tlv65NTBz83BJBJvmLVHHuKJVaDOBvJJykiT5TR++em5nfAySPccDZDa4oSrn8A==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.cjs" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/browserslist": { + "version": "4.28.2", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz", + "integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "peer": true, + "dependencies": { + "baseline-browser-mapping": "^2.10.12", + "caniuse-lite": "^1.0.30001782", + "electron-to-chromium": "^1.5.328", + "node-releases": "^2.0.36", + "update-browserslist-db": "^1.2.3" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001787", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001787.tgz", + "integrity": "sha512-mNcrMN9KeI68u7muanUpEejSLghOKlVhRqS/Za2IeyGllJ9I9otGpR9g3nsw7n4W378TE/LyIteA0+/FOZm4Kg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "license": "MIT" + }, + "node_modules/cookie": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.1.1.tgz", + "integrity": "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ==", + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/csstype": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", + "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/detect-libc": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", + "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=8" + } + }, + "node_modules/electron-to-chromium": { + "version": "1.5.335", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.335.tgz", + "integrity": "sha512-q9n5T4BR4Xwa2cwbrwcsDJtHD/enpQ5S1xF1IAtdqf5AAgqDFmR/aakqH3ChFdqd/QXJhS3rnnXFtexU7rax6Q==", + "dev": true, + "license": "ISC" + }, + "node_modules/enhanced-resolve": { + "version": "5.20.1", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.20.1.tgz", + "integrity": "sha512-Qohcme7V1inbAfvjItgw0EaxVX5q2rdVEZHRBrEQdRZTssLDGsL8Lwrznl8oQ/6kuTJONLaDcGjkNP247XEhcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "graceful-fs": "^4.2.4", + "tapable": "^2.3.0" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/esbuild": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz", + "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.25.12", + "@esbuild/android-arm": "0.25.12", + "@esbuild/android-arm64": "0.25.12", + "@esbuild/android-x64": "0.25.12", + "@esbuild/darwin-arm64": "0.25.12", + "@esbuild/darwin-x64": "0.25.12", + "@esbuild/freebsd-arm64": "0.25.12", + "@esbuild/freebsd-x64": "0.25.12", + "@esbuild/linux-arm": "0.25.12", + "@esbuild/linux-arm64": "0.25.12", + "@esbuild/linux-ia32": "0.25.12", + "@esbuild/linux-loong64": "0.25.12", + "@esbuild/linux-mips64el": "0.25.12", + "@esbuild/linux-ppc64": "0.25.12", + "@esbuild/linux-riscv64": "0.25.12", + "@esbuild/linux-s390x": "0.25.12", + "@esbuild/linux-x64": "0.25.12", + "@esbuild/netbsd-arm64": "0.25.12", + "@esbuild/netbsd-x64": "0.25.12", + "@esbuild/openbsd-arm64": "0.25.12", + "@esbuild/openbsd-x64": "0.25.12", + "@esbuild/openharmony-arm64": "0.25.12", + "@esbuild/sunos-x64": "0.25.12", + "@esbuild/win32-arm64": "0.25.12", + "@esbuild/win32-ia32": "0.25.12", + "@esbuild/win32-x64": "0.25.12" + } + }, + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/jiti": { + "version": "2.6.1", + "resolved": "https://registry.npmjs.org/jiti/-/jiti-2.6.1.tgz", + "integrity": "sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ==", + "dev": true, + "license": "MIT", + "bin": { + "jiti": "lib/jiti-cli.mjs" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "license": "MIT" + }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/lightningcss": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz", + "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==", + "dev": true, + "license": "MPL-2.0", + "dependencies": { + "detect-libc": "^2.0.3" + }, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + }, + "optionalDependencies": { + "lightningcss-android-arm64": "1.32.0", + "lightningcss-darwin-arm64": "1.32.0", + "lightningcss-darwin-x64": "1.32.0", + "lightningcss-freebsd-x64": "1.32.0", + "lightningcss-linux-arm-gnueabihf": "1.32.0", + "lightningcss-linux-arm64-gnu": "1.32.0", + "lightningcss-linux-arm64-musl": "1.32.0", + "lightningcss-linux-x64-gnu": "1.32.0", + "lightningcss-linux-x64-musl": "1.32.0", + "lightningcss-win32-arm64-msvc": "1.32.0", + "lightningcss-win32-x64-msvc": "1.32.0" + } + }, + "node_modules/lightningcss-android-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz", + "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz", + "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz", + "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-freebsd-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz", + "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm-gnueabihf": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz", + "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz", + "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz", + "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz", + "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz", + "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-arm64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz", + "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-x64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz", + "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "license": "MIT", + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/magic-string": { + "version": "0.30.21", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", + "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.5" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/nanoid": { + "version": "3.3.11", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz", + "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/node-releases": { + "version": "2.0.37", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.37.tgz", + "integrity": "sha512-1h5gKZCF+pO/o3Iqt5Jp7wc9rH3eJJ0+nh/CIoiRwjRxde/hAHyLPXYN4V3CqKAbiZPSeJFSWHmJsbkicta0Eg==", + "dev": true, + "license": "MIT" + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/postcss": { + "version": "8.5.9", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.9.tgz", + "integrity": "sha512-7a70Nsot+EMX9fFU3064K/kdHWZqGVY+BADLyXc8Dfv+mTLLVl6JzJpPaCZ2kQL9gIJvKXSLMHhqdRRjwQeFtw==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "nanoid": "^3.3.11", + "picocolors": "^1.1.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/react": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz", + "integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==", + "license": "MIT", + "peer": true, + "dependencies": { + "loose-envify": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz", + "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==", + "license": "MIT", + "peer": true, + "dependencies": { + "loose-envify": "^1.1.0", + "scheduler": "^0.23.2" + }, + "peerDependencies": { + "react": "^18.3.1" + } + }, + "node_modules/react-refresh": { + "version": "0.17.0", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.17.0.tgz", + "integrity": "sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-router": { + "version": "7.14.0", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-7.14.0.tgz", + "integrity": "sha512-m/xR9N4LQLmAS0ZhkY2nkPA1N7gQ5TUVa5n8TgANuDTARbn1gt+zLPXEm7W0XDTbrQ2AJSJKhoa6yx1D8BcpxQ==", + "license": "MIT", + "dependencies": { + "cookie": "^1.0.1", + "set-cookie-parser": "^2.6.0" + }, + "engines": { + "node": ">=20.0.0" + }, + "peerDependencies": { + "react": ">=18", + "react-dom": ">=18" + }, + "peerDependenciesMeta": { + "react-dom": { + "optional": true + } + } + }, + "node_modules/react-router-dom": { + "version": "7.14.0", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-7.14.0.tgz", + "integrity": "sha512-2G3ajSVSZMEtmTjIklRWlNvo8wICEpLihfD/0YMDxbWK2UyP5EGfnoIn9AIQGnF3G/FX0MRbHXdFcD+rL1ZreQ==", + "license": "MIT", + "dependencies": { + "react-router": "7.14.0" + }, + "engines": { + "node": ">=20.0.0" + }, + "peerDependencies": { + "react": ">=18", + "react-dom": ">=18" + } + }, + "node_modules/rollup": { + "version": "4.60.1", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.60.1.tgz", + "integrity": "sha512-VmtB2rFU/GroZ4oL8+ZqXgSA38O6GR8KSIvWmEFv63pQ0G6KaBH9s07PO8XTXP4vI+3UJUEypOfjkGfmSBBR0w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "1.0.8" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.60.1", + "@rollup/rollup-android-arm64": "4.60.1", + "@rollup/rollup-darwin-arm64": "4.60.1", + "@rollup/rollup-darwin-x64": "4.60.1", + "@rollup/rollup-freebsd-arm64": "4.60.1", + "@rollup/rollup-freebsd-x64": "4.60.1", + "@rollup/rollup-linux-arm-gnueabihf": "4.60.1", + "@rollup/rollup-linux-arm-musleabihf": "4.60.1", + "@rollup/rollup-linux-arm64-gnu": "4.60.1", + "@rollup/rollup-linux-arm64-musl": "4.60.1", + "@rollup/rollup-linux-loong64-gnu": "4.60.1", + "@rollup/rollup-linux-loong64-musl": "4.60.1", + "@rollup/rollup-linux-ppc64-gnu": "4.60.1", + "@rollup/rollup-linux-ppc64-musl": "4.60.1", + "@rollup/rollup-linux-riscv64-gnu": "4.60.1", + "@rollup/rollup-linux-riscv64-musl": "4.60.1", + "@rollup/rollup-linux-s390x-gnu": "4.60.1", + "@rollup/rollup-linux-x64-gnu": "4.60.1", + "@rollup/rollup-linux-x64-musl": "4.60.1", + "@rollup/rollup-openbsd-x64": "4.60.1", + "@rollup/rollup-openharmony-arm64": "4.60.1", + "@rollup/rollup-win32-arm64-msvc": "4.60.1", + "@rollup/rollup-win32-ia32-msvc": "4.60.1", + "@rollup/rollup-win32-x64-gnu": "4.60.1", + "@rollup/rollup-win32-x64-msvc": "4.60.1", + "fsevents": "~2.3.2" + } + }, + "node_modules/scheduler": { + "version": "0.23.2", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz", + "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==", + "license": "MIT", + "dependencies": { + "loose-envify": "^1.1.0" + } + }, + "node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/set-cookie-parser": { + "version": "2.7.2", + "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.2.tgz", + "integrity": "sha512-oeM1lpU/UvhTxw+g3cIfxXHyJRc/uidd3yK1P242gzHds0udQBYzs3y8j4gCCW+ZJ7ad0yctld8RYO+bdurlvw==", + "license": "MIT" + }, + "node_modules/source-map-js": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", + "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/tailwindcss": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-4.2.2.tgz", + "integrity": "sha512-KWBIxs1Xb6NoLdMVqhbhgwZf2PGBpPEiwOqgI4pFIYbNTfBXiKYyWoTsXgBQ9WFg/OlhnvHaY+AEpW7wSmFo2Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/tapable": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.2.tgz", + "integrity": "sha512-1MOpMXuhGzGL5TTCZFItxCc0AARf1EZFQkGqMm7ERKj8+Hgr5oLvJOVFcC+lRmR8hCe2S3jC4T5D7Vg/d7/fhA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/tinyglobby": { + "version": "0.2.16", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz", + "integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.4" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, + "node_modules/typescript": { + "version": "5.9.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", + "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz", + "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/vite": { + "version": "6.4.2", + "resolved": "https://registry.npmjs.org/vite/-/vite-6.4.2.tgz", + "integrity": "sha512-2N/55r4JDJ4gdrCvGgINMy+HH3iRpNIz8K6SFwVsA+JbQScLiC+clmAxBgwiSPgcG9U15QmvqCGWzMbqda5zGQ==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "esbuild": "^0.25.0", + "fdir": "^6.4.4", + "picomatch": "^4.0.2", + "postcss": "^8.5.3", + "rollup": "^4.34.9", + "tinyglobby": "^0.2.13" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^18.0.0 || ^20.0.0 || >=22.0.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", + "jiti": ">=1.21.0", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "sass-embedded": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true + } + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..c6da43e --- /dev/null +++ b/package.json @@ -0,0 +1,29 @@ +{ + "name": "orchai", + "private": true, + "version": "0.1.0", + "type": "module", + "scripts": { + "dev": "vite", + "build": "tsc && vite build", + "preview": "vite preview", + "tauri": "tauri" + }, + "dependencies": { + "@tauri-apps/api": "^2.10.1", + "@tauri-apps/plugin-dialog": "^2.7.0", + "react": "^18.3.1", + "react-dom": "^18.3.1", + "react-router-dom": "^7.14.0" + }, + "devDependencies": { + "@tailwindcss/vite": "^4.2.2", + "@tauri-apps/cli": "^2", + "@types/react": "^18.3.1", + "@types/react-dom": "^18.3.1", + "@vitejs/plugin-react": "^4.3.1", + "tailwindcss": "^4.2.2", + "typescript": "^5.6.3", + "vite": "^6.0.3" + } +} diff --git a/src-tauri/Cargo.lock b/src-tauri/Cargo.lock new file mode 100644 index 0000000..e030ba3 --- /dev/null +++ b/src-tauri/Cargo.lock @@ -0,0 +1,5733 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 4 + +[[package]] +name = "adler2" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "320119579fcad9c21884f5c4861d16174d0e06250625266f50fe6898340abefa" + +[[package]] +name = "aead" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d122413f284cf2d62fb1b7db97e02edb8cda96d769b16e443a4f6195e35662b0" +dependencies = [ + "crypto-common", + "generic-array", +] + +[[package]] +name = "aes" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0" +dependencies = [ + "cfg-if", + "cipher", + "cpufeatures", +] + +[[package]] +name = "aes-gcm" +version = "0.10.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "831010a0f742e1209b3bcea8fab6a8e149051ba6099432c8cb2cc117dec3ead1" +dependencies = [ + "aead", + "aes", + "cipher", + "ctr", + "ghash", + "subtle", +] + +[[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 = "alloc-no-stdlib" +version = "2.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc7bb162ec39d46ab1ca8c77bf72e890535becd1751bb45f64c597edb4c8c6b3" + +[[package]] +name = "alloc-stdlib" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94fb8275041c72129eb51b7d0322c29b8387a0386127718b096429201a5d6ece" +dependencies = [ + "alloc-no-stdlib", +] + +[[package]] +name = "android_system_properties" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311" +dependencies = [ + "libc", +] + +[[package]] +name = "anyhow" +version = "1.0.102" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f202df86484c868dbad7eaa557ef785d5c66295e41b460ef922eca0723b842c" + +[[package]] +name = "atk" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "241b621213072e993be4f6f3a9e4b45f65b7e6faad43001be957184b7bb1824b" +dependencies = [ + "atk-sys", + "glib", + "libc", +] + +[[package]] +name = "atk-sys" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c5e48b684b0ca77d2bbadeef17424c2ea3c897d44d566a1617e7e8f30614d086" +dependencies = [ + "glib-sys", + "gobject-sys", + "libc", + "system-deps", +] + +[[package]] +name = "atomic-waker" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0" + +[[package]] +name = "autocfg" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c08606f8c3cbf4ce6ec8e28fb0014a2c086708fe954eaa885384a6165172e7e8" + +[[package]] +name = "base64" +version = "0.21.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" + +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + +[[package]] +name = "bit-set" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08807e080ed7f9d5433fa9b275196cfc35414f66a0c79d864dc51a0d825231a3" +dependencies = [ + "bit-vec", +] + +[[package]] +name = "bit-vec" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e764a1d40d510daf35e07be9eb06e75770908c27d411ee6c92109c9840eaaf7" + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + +[[package]] +name = "bitflags" +version = "2.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "843867be96c8daad0d758b57df9392b6d8d271134fce549de6ce169ff98a92af" +dependencies = [ + "serde_core", +] + +[[package]] +name = "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 = "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.20.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d20789868f4b01b2f2caec9f5c4e0213b41e3e5702a50157d699ae31ced2fcb" + +[[package]] +name = "bytemuck" +version = "1.25.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8efb64bd706a16a1bdde310ae86b351e4d21550d98d056f22f8a7f7a2183fec" + +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" + +[[package]] +name = "bytes" +version = "1.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e748733b7cbc798e1434b6ac524f0c1ff2ab456fe201501e6497c8417a4fc33" +dependencies = [ + "serde", +] + +[[package]] +name = "cairo-rs" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ca26ef0159422fb77631dc9d17b102f253b876fe1586b03b803e63a309b4ee2" +dependencies = [ + "bitflags 2.11.0", + "cairo-sys-rs", + "glib", + "libc", + "once_cell", + "thiserror 1.0.69", +] + +[[package]] +name = "cairo-sys-rs" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "685c9fa8e590b8b3d678873528d83411db17242a73fccaed827770ea0fedda51" +dependencies = [ + "glib-sys", + "libc", + "system-deps", +] + +[[package]] +name = "camino" +version = "1.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e629a66d692cb9ff1a1c664e41771b3dcaf961985a9774c0eb0bd1b51cf60a48" +dependencies = [ + "serde_core", +] + +[[package]] +name = "cargo-platform" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e35af189006b9c0f00a064685c727031e3ed2d8020f7ba284d78cc2671bd36ea" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo_metadata" +version = "0.19.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd5eb614ed4c27c5d706420e4320fbe3216ab31fa1c33cd8246ac36dae4479ba" +dependencies = [ + "camino", + "cargo-platform", + "semver", + "serde", + "serde_json", + "thiserror 2.0.18", +] + +[[package]] +name = "cargo_toml" +version = "0.22.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "374b7c592d9c00c1f4972ea58390ac6b18cbb6ab79011f3bdc90a0b82ca06b77" +dependencies = [ + "serde", + "toml 0.9.12+spec-1.1.0", +] + +[[package]] +name = "cc" +version = "1.2.60" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43c5703da9466b66a946814e1adf53ea2c90f10063b86290cc9eb67ce3478a20" +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 = "chrono" +version = "0.4.44" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c673075a2e0e5f4a1dde27ce9dee1ea4558c7ffe648f576438a20ca1d2acc4b0" +dependencies = [ + "iana-time-zone", + "js-sys", + "num-traits", + "serde", + "wasm-bindgen", + "windows-link 0.2.1", +] + +[[package]] +name = "cipher" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad" +dependencies = [ + "crypto-common", + "inout", +] + +[[package]] +name = "combine" +version = "4.6.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd" +dependencies = [ + "bytes", + "memchr", +] + +[[package]] +name = "convert_case" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" + +[[package]] +name = "cookie" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ddef33a339a91ea89fb53151bd0a4689cfce27055c291dfa69945475d22c747" +dependencies = [ + "time", + "version_check", +] + +[[package]] +name = "core-foundation" +version = "0.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.25.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "064badf302c3194842cf2c5d61f56cc88e54a759313879cdf03abdd27d0c3b97" +dependencies = [ + "bitflags 2.11.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.11.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 = "crc32fast" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9481c1c90cbf2ac953f07c8d4a58aa3945c425b7185c9154d67a65e4230da511" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "crossbeam-channel" +version = "0.5.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "82b8f8f868b36967f9606790d1903570de9ceaf870a7bf9fbbd3016d636a2cb2" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28" + +[[package]] +name = "crypto-common" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78c8292055d1c1df0cce5d180393dc8cce0abec0a7102adb6c7b1eef6016d60a" +dependencies = [ + "generic-array", + "rand_core 0.6.4", + "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" +version = "0.36.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dae61cf9c0abb83bd659dab65b7e4e38d8236824c85f0f804f173567bda257d2" +dependencies = [ + "cssparser-macros", + "dtoa-short", + "itoa", + "phf 0.13.1", + "smallvec", +] + +[[package]] +name = "cssparser-macros" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13b588ba4ac1a99f7f2964d24b3d896ddc6bf847ee3855dbd4366f058cfcd331" +dependencies = [ + "quote", + "syn 2.0.117", +] + +[[package]] +name = "ctor" +version = "0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32a2785755761f3ddc1492979ce1e48d2c00d09311c39e4466429188f3dd6501" +dependencies = [ + "quote", + "syn 2.0.117", +] + +[[package]] +name = "ctr" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0369ee1ad671834580515889b80f2ea915f23b8be8d0daa4bbaf2ac5c7590835" +dependencies = [ + "cipher", +] + +[[package]] +name = "darling" +version = "0.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "25ae13da2f202d56bd7f91c25fba009e7717a1e4a1cc98a76d844b65ae912e9d" +dependencies = [ + "darling_core", + "darling_macro", +] + +[[package]] +name = "darling_core" +version = "0.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9865a50f7c335f53564bb694ef660825eb8610e0a53d3e11bf1b0d3df31e03b0" +dependencies = [ + "ident_case", + "proc-macro2", + "quote", + "strsim", + "syn 2.0.117", +] + +[[package]] +name = "darling_macro" +version = "0.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3984ec7bd6cfa798e62b4a642426a5be0e68f9401cfc2a01e3fa9ea2fcdb8d" +dependencies = [ + "darling_core", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "deranged" +version = "0.5.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7cd812cc2bc1d69d4764bd80df88b4317eaef9e773c75226407d9bc0876b211c" +dependencies = [ + "powerfmt", + "serde_core", +] + +[[package]] +name = "derive_more" +version = "0.99.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6edb4b64a43d977b8e99788fe3a04d483834fba1215a7e02caa415b626497f7f" +dependencies = [ + "convert_case", + "proc-macro2", + "quote", + "rustc_version", + "syn 2.0.117", +] + +[[package]] +name = "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.117", +] + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer", + "crypto-common", +] + +[[package]] +name = "dirs" +version = "5.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44c45a9d03d6676652bcb5e724c7e988de1acad23a711b5217ab9cbecbec2225" +dependencies = [ + "dirs-sys 0.4.1", +] + +[[package]] +name = "dirs" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3e8aa94d75141228480295a7d0e7feb620b1a5ad9f12bc40be62411e38cce4e" +dependencies = [ + "dirs-sys 0.5.0", +] + +[[package]] +name = "dirs-sys" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "520f05a5cbd335fae5a99ff7a6ab8627577660ee5cfd6a94a6a929b52ff0321c" +dependencies = [ + "libc", + "option-ext", + "redox_users 0.4.6", + "windows-sys 0.48.0", +] + +[[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 0.5.2", + "windows-sys 0.61.2", +] + +[[package]] +name = "dispatch2" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e0e367e4e7da84520dedcac1901e4da967309406d1e51017ae1abfb97adbd38" +dependencies = [ + "bitflags 2.11.0", + "block2", + "libc", + "objc2", +] + +[[package]] +name = "displaydoc" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "dlopen2" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e2c5bd4158e66d1e215c49b837e11d62f3267b30c92f1d171c4d3105e3dc4d4" +dependencies = [ + "dlopen2_derive", + "libc", + "once_cell", + "winapi", +] + +[[package]] +name = "dlopen2_derive" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fbbb781877580993a8707ec48672673ec7b81eeba04cfd2310bd28c08e47c8f" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "dom_query" +version = "0.27.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "521e380c0c8afb8d9a1e83a1822ee03556fc3e3e7dbc1fd30be14e37f9cb3f89" +dependencies = [ + "bit-set", + "cssparser 0.36.0", + "foldhash 0.2.0", + "html5ever 0.38.0", + "precomputed-hash", + "selectors 0.36.1", + "tendril 0.5.0", +] + +[[package]] +name = "dpi" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d8b14ccef22fc6f5a8f4d7d768562a182c04ce9a3b3157b91390b52ddfdf1a76" +dependencies = [ + "serde", +] + +[[package]] +name = "dtoa" +version = "1.0.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c3cf4824e2d5f025c7b531afcb2325364084a16806f6d47fbc1f5fbd9960590" + +[[package]] +name = "dtoa-short" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cd1511a7b6a56299bd043a9c167a6d2bfb37bf84a6dfceaba651168adfb43c87" +dependencies = [ + "dtoa", +] + +[[package]] +name = "dunce" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92773504d58c093f6de2459af4af33faa518c13451eb8f2b5698ed3d36e7c813" + +[[package]] +name = "dyn-clone" +version = "1.0.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d0881ea181b1df73ff77ffaaf9c7544ecc11e82fba9b5f27b262a3c73a332555" + +[[package]] +name = "embed-resource" +version = "3.0.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63a1d0de4f2249aa0ff5884d7080814f446bb241a559af6c170a41e878ed2d45" +dependencies = [ + "cc", + "memchr", + "rustc_version", + "toml 0.9.12+spec-1.1.0", + "vswhom", + "winreg", +] + +[[package]] +name = "embed_plist" +version = "1.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ef6b89e5b37196644d8796de5268852ff179b44e96276cf4290264843743bb7" + +[[package]] +name = "encoding_rs" +version = "0.8.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75030f3c4f45dafd7586dd6780965a8c7e8e285a5ecb86713e63a79c5b2766f3" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "equivalent" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f" + +[[package]] +name = "erased-serde" +version = "0.4.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d2add8a07dd6a8d93ff627029c51de145e12686fbc36ecb298ac22e74cf02dec" +dependencies = [ + "serde", + "serde_core", + "typeid", +] + +[[package]] +name = "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 = "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.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f1f227452a390804cdb637b74a86990f2a7d7ba4b7d5693aac9b4dd6defd8d6" + +[[package]] +name = "fdeflate" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e6853b52649d4ac5c0bd02320cddc5ba956bdb407c4b75a2c6b75bf51500f8c" +dependencies = [ + "simd-adler32", +] + +[[package]] +name = "field-offset" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38e2275cc4e4fc009b0669731a1e5ab7ebf11f469eaede2bab9309a5b4d6057f" +dependencies = [ + "memoffset", + "rustc_version", +] + +[[package]] +name = "find-msvc-tools" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5baebc0774151f905a1a2cc41989300b1e6fbb29aff0ceffa1064fdd3088d582" + +[[package]] +name = "flate2" +version = "1.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "843fba2746e448b37e26a819579957415c8cef339bf08564fe8b7ddbd959573c" +dependencies = [ + "crc32fast", + "miniz_oxide", +] + +[[package]] +name = "fnv" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" + +[[package]] +name = "foldhash" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9c4f5dac5e15c24eb999c26181a6ca40b39fe946cbe4c263c7209467bc83af2" + +[[package]] +name = "foldhash" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77ce24cb58228fbb8aa041425bb1050850ac19177686ea6e0f41a70416f56fdb" + +[[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.117", +] + +[[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 = "futf" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df420e2e84819663797d1ec6544b13c5be84629e7bb00dc960d6917db2987843" +dependencies = [ + "mac", + "new_debug_unreachable", +] + +[[package]] +name = "futures-channel" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07bbe89c50d7a535e539b8c17bc0b49bdb77747034daa8087407d655f3f7cc1d" +dependencies = [ + "futures-core", +] + +[[package]] +name = "futures-core" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e3450815272ef58cec6d564423f6e755e25379b217b0bc688e295ba24df6b1d" + +[[package]] +name = "futures-executor" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf29c38818342a3b26b5b923639e7b1f4a61fc5e76102d4b1981c6dc7a7579d" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-io" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cecba35d7ad927e23624b22ad55235f2239cfa44fd10428eecbeba6d6a717718" + +[[package]] +name = "futures-macro" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e835b70203e41293343137df5c0664546da5745f82ec9b84d40be8336958447b" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "futures-sink" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c39754e157331b013978ec91992bde1ac089843443c49cbc7f46150b0fad0893" + +[[package]] +name = "futures-task" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "037711b3d59c33004d3856fbdc83b99d4ff37a24768fa1be9ce3538a1cde4393" + +[[package]] +name = "futures-util" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "389ca41296e6190b48053de0321d02a77f32f8a5d2461dd38762c0593805c6d6" +dependencies = [ + "futures-core", + "futures-io", + "futures-macro", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "slab", +] + +[[package]] +name = "fxhash" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c31b6d751ae2c7f11320402d34e41349dd1016f8d5d45e48c4312bc8625af50c" +dependencies = [ + "byteorder", +] + +[[package]] +name = "gdk" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9f245958c627ac99d8e529166f9823fb3b838d1d41fd2b297af3075093c2691" +dependencies = [ + "cairo-rs", + "gdk-pixbuf", + "gdk-sys", + "gio", + "glib", + "libc", + "pango", +] + +[[package]] +name = "gdk-pixbuf" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50e1f5f1b0bfb830d6ccc8066d18db35c487b1b2b1e8589b5dfe9f07e8defaec" +dependencies = [ + "gdk-pixbuf-sys", + "gio", + "glib", + "libc", + "once_cell", +] + +[[package]] +name = "gdk-pixbuf-sys" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f9839ea644ed9c97a34d129ad56d38a25e6756f99f3a88e15cd39c20629caf7" +dependencies = [ + "gio-sys", + "glib-sys", + "gobject-sys", + "libc", + "system-deps", +] + +[[package]] +name = "gdk-sys" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c2d13f38594ac1e66619e188c6d5a1adb98d11b2fcf7894fc416ad76aa2f3f7" +dependencies = [ + "cairo-sys-rs", + "gdk-pixbuf-sys", + "gio-sys", + "glib-sys", + "gobject-sys", + "libc", + "pango-sys", + "pkg-config", + "system-deps", +] + +[[package]] +name = "gdkwayland-sys" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "140071d506d223f7572b9f09b5e155afbd77428cd5cc7af8f2694c41d98dfe69" +dependencies = [ + "gdk-sys", + "glib-sys", + "gobject-sys", + "libc", + "pkg-config", + "system-deps", +] + +[[package]] +name = "gdkx11" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3caa00e14351bebbc8183b3c36690327eb77c49abc2268dd4bd36b856db3fbfe" +dependencies = [ + "gdk", + "gdkx11-sys", + "gio", + "glib", + "libc", + "x11", +] + +[[package]] +name = "gdkx11-sys" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e2e7445fe01ac26f11601db260dd8608fe172514eb63b3b5e261ea6b0f4428d" +dependencies = [ + "gdk-sys", + "glib-sys", + "libc", + "system-deps", + "x11", +] + +[[package]] +name = "generic-array" +version = "0.14.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a" +dependencies = [ + "typenum", + "version_check", +] + +[[package]] +name = "getrandom" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" +dependencies = [ + "cfg-if", + "libc", + "wasi 0.9.0+wasi-snapshot-preview1", +] + +[[package]] +name = "getrandom" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff2abc00be7fca6ebc474524697ae276ad847ad0a6b3faa4bcb027e9a4614ad0" +dependencies = [ + "cfg-if", + "libc", + "wasi 0.11.1+wasi-snapshot-preview1", +] + +[[package]] +name = "getrandom" +version = "0.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "899def5c37c4fd7b2664648c28120ecec138e4d395b459e5ca34f9cce2dd77fd" +dependencies = [ + "cfg-if", + "libc", + "r-efi 5.3.0", + "wasip2", +] + +[[package]] +name = "getrandom" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0de51e6874e94e7bf76d726fc5d13ba782deca734ff60d5bb2fb2607c7406555" +dependencies = [ + "cfg-if", + "libc", + "r-efi 6.0.0", + "wasip2", + "wasip3", +] + +[[package]] +name = "ghash" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0d8a4362ccb29cb0b265253fb0a2728f592895ee6854fd9bc13f2ffda266ff1" +dependencies = [ + "opaque-debug", + "polyval", +] + +[[package]] +name = "gio" +version = "0.18.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d4fc8f532f87b79cbc51a79748f16a6828fb784be93145a322fa14d06d354c73" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-util", + "gio-sys", + "glib", + "libc", + "once_cell", + "pin-project-lite", + "smallvec", + "thiserror 1.0.69", +] + +[[package]] +name = "gio-sys" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "37566df850baf5e4cb0dfb78af2e4b9898d817ed9263d1090a2df958c64737d2" +dependencies = [ + "glib-sys", + "gobject-sys", + "libc", + "system-deps", + "winapi", +] + +[[package]] +name = "glib" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "233daaf6e83ae6a12a52055f568f9d7cf4671dabb78ff9560ab6da230ce00ee5" +dependencies = [ + "bitflags 2.11.0", + "futures-channel", + "futures-core", + "futures-executor", + "futures-task", + "futures-util", + "gio-sys", + "glib-macros", + "glib-sys", + "gobject-sys", + "libc", + "memchr", + "once_cell", + "smallvec", + "thiserror 1.0.69", +] + +[[package]] +name = "glib-macros" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bb0228f477c0900c880fd78c8759b95c7636dbd7842707f49e132378aa2acdc" +dependencies = [ + "heck 0.4.1", + "proc-macro-crate 2.0.2", + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "glib-sys" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "063ce2eb6a8d0ea93d2bf8ba1957e78dbab6be1c2220dd3daca57d5a9d869898" +dependencies = [ + "libc", + "system-deps", +] + +[[package]] +name = "glob" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0cc23270f6e1808e30a928bdc84dea0b9b4136a8bc82338574f23baf47bbd280" + +[[package]] +name = "gobject-sys" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0850127b514d1c4a4654ead6dedadb18198999985908e6ffe4436f53c785ce44" +dependencies = [ + "glib-sys", + "libc", + "system-deps", +] + +[[package]] +name = "gtk" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd56fb197bfc42bd5d2751f4f017d44ff59fbb58140c6b49f9b3b2bdab08506a" +dependencies = [ + "atk", + "cairo-rs", + "field-offset", + "futures-channel", + "gdk", + "gdk-pixbuf", + "gio", + "glib", + "gtk-sys", + "gtk3-macros", + "libc", + "pango", + "pkg-config", +] + +[[package]] +name = "gtk-sys" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f29a1c21c59553eb7dd40e918be54dccd60c52b049b75119d5d96ce6b624414" +dependencies = [ + "atk-sys", + "cairo-sys-rs", + "gdk-pixbuf-sys", + "gdk-sys", + "gio-sys", + "glib-sys", + "gobject-sys", + "libc", + "pango-sys", + "system-deps", +] + +[[package]] +name = "gtk3-macros" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52ff3c5b21f14f0736fed6dcfc0bfb4225ebf5725f3c0209edeec181e4d73e9d" +dependencies = [ + "proc-macro-crate 1.3.1", + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "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.14.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" + +[[package]] +name = "hashbrown" +version = "0.14.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" +dependencies = [ + "ahash", +] + +[[package]] +name = "hashbrown" +version = "0.15.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9229cfe53dfd69f0609a49f65461bd93001ea1ef889cd5529dd176593f5338a1" +dependencies = [ + "foldhash 0.1.5", +] + +[[package]] +name = "hashbrown" +version = "0.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f467dd6dccf739c208452f8014c75c18bb8301b050ad1cfb27153803edb0f51" + +[[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 = "heck" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8" + +[[package]] +name = "heck" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea" + +[[package]] +name = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "html5ever" +version = "0.29.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b7410cae13cbc75623c98ac4cbfd1f0bedddf3227afc24f370cf0f50a44a11c" +dependencies = [ + "log", + "mac", + "markup5ever 0.14.1", + "match_token", +] + +[[package]] +name = "html5ever" +version = "0.38.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1054432bae2f14e0061e33d23402fbaa67a921d319d56adc6bcf887ddad1cbc2" +dependencies = [ + "log", + "markup5ever 0.38.0", +] + +[[package]] +name = "http" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3ba2a386d7f85a81f119ad7498ebe444d2e22c2af0b86b069416ace48b3311a" +dependencies = [ + "bytes", + "itoa", +] + +[[package]] +name = "http-body" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" +dependencies = [ + "bytes", + "http", +] + +[[package]] +name = "http-body-util" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b021d93e26becf5dc7e1b75b1bed1fd93124b374ceb73f43d4d4eafec896a64a" +dependencies = [ + "bytes", + "futures-core", + "http", + "http-body", + "pin-project-lite", +] + +[[package]] +name = "httparse" +version = "1.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6dbf3de79e51f3d586ab4cb9d5c3e2c14aa28ed23d180cf89b4df0454a69cc87" + +[[package]] +name = "hyper" +version = "1.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6299f016b246a94207e63da54dbe807655bf9e00044f73ded42c3ac5305fbcca" +dependencies = [ + "atomic-waker", + "bytes", + "futures-channel", + "futures-core", + "h2", + "http", + "http-body", + "httparse", + "itoa", + "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-rustls" +version = "0.27.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2b52f86d1d4bc0d6b4e6826d960b1b333217e07d36b882dca570a5e1c48895b" +dependencies = [ + "http", + "hyper", + "hyper-util", + "rustls", + "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.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2984d1cd16c883d7935b9e07e44071dca8d917fd52ecc02c04d5fa0b5a3f191c" +dependencies = [ + "displaydoc", + "potential_utf", + "utf8_iter", + "yoke", + "zerofrom", + "zerovec", +] + +[[package]] +name = "icu_locale_core" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92219b62b3e2b4d88ac5119f8904c10f8f61bf7e95b640d25ba3075e6cac2c29" +dependencies = [ + "displaydoc", + "litemap", + "tinystr", + "writeable", + "zerovec", +] + +[[package]] +name = "icu_normalizer" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c56e5ee99d6e3d33bd91c5d85458b6005a22140021cc324cea84dd0e72cff3b4" +dependencies = [ + "icu_collections", + "icu_normalizer_data", + "icu_properties", + "icu_provider", + "smallvec", + "zerovec", +] + +[[package]] +name = "icu_normalizer_data" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da3be0ae77ea334f4da67c12f149704f19f81d1adf7c51cf482943e84a2bad38" + +[[package]] +name = "icu_properties" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bee3b67d0ea5c2cca5003417989af8996f8604e34fb9ddf96208a033901e70de" +dependencies = [ + "icu_collections", + "icu_locale_core", + "icu_properties_data", + "icu_provider", + "zerotrie", + "zerovec", +] + +[[package]] +name = "icu_properties_data" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e2bbb201e0c04f7b4b3e14382af113e17ba4f63e2c9d2ee626b720cbce54a14" + +[[package]] +name = "icu_provider" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "139c4cf31c8b5f33d7e199446eff9c1e02decfc2f0eec2c8d71f65befa45b421" +dependencies = [ + "displaydoc", + "icu_locale_core", + "writeable", + "yoke", + "zerofrom", + "zerotrie", + "zerovec", +] + +[[package]] +name = "id-arena" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d3067d79b975e8844ca9eb072e16b31c3c1c36928edf9c6789548c524d0d954" + +[[package]] +name = "ident_case" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" + +[[package]] +name = "idna" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b0875f23caa03898994f6ddc501886a45c7d3d62d04d2d90788d47be1b1e4de" +dependencies = [ + "idna_adapter", + "smallvec", + "utf8_iter", +] + +[[package]] +name = "idna_adapter" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3acae9609540aa318d1bc588455225fb2085b9ed0c4f6bd0d9d5bcd86f1a0344" +dependencies = [ + "icu_normalizer", + "icu_properties", +] + +[[package]] +name = "indexmap" +version = "1.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd070e393353796e801d209ad339e89596eb4c8d430d18ede6a1cced8fafbd99" +dependencies = [ + "autocfg", + "hashbrown 0.12.3", + "serde", +] + +[[package]] +name = "indexmap" +version = "2.14.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d466e9454f08e4a911e14806c24e16fba1b4c121d1ea474396f396069cf949d9" +dependencies = [ + "equivalent", + "hashbrown 0.17.0", + "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 = "inout" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "879f10e63c20629ecabbb64a8010319738c66a5cd0c29b02d63d272b03751d01" +dependencies = [ + "generic-array", +] + +[[package]] +name = "ipnet" +version = "2.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d98f6fed1fde3f8c21bc40a1abb88dd75e67924f9cffc3ef95607bad8017f8e2" + +[[package]] +name = "iri-string" +version = "0.7.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "25e659a4bb38e810ebc252e53b5814ff908a8c58c2a9ce2fae1bbec24cbf4e20" +dependencies = [ + "memchr", + "serde", +] + +[[package]] +name = "itoa" +version = "1.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f42a60cbdf9a97f5d2305f08a87dc4e09308d1276d28c869c684d7777685682" + +[[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 0.3.1", + "log", + "thiserror 1.0.69", + "walkdir", + "windows-sys 0.45.0", +] + +[[package]] +name = "jni-sys" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41a652e1f9b6e0275df1f15b32661cf0d4b78d4d87ddec5e0c3c20f097433258" +dependencies = [ + "jni-sys 0.4.1", +] + +[[package]] +name = "jni-sys" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6377a88cb3910bee9b0fa88d4f42e1d2da8e79915598f65fb0c7ee14c878af2" +dependencies = [ + "jni-sys-macros", +] + +[[package]] +name = "jni-sys-macros" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38c0b942f458fe50cdac086d2f946512305e5631e720728f2a61aabcd47a6264" +dependencies = [ + "quote", + "syn 2.0.117", +] + +[[package]] +name = "js-sys" +version = "0.3.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2964e92d1d9dc3364cae4d718d93f227e3abb088e747d92e0395bfdedf1c12ca" +dependencies = [ + "cfg-if", + "futures-util", + "once_cell", + "wasm-bindgen", +] + +[[package]] +name = "json-patch" +version = "3.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "863726d7afb6bc2590eeff7135d923545e5e964f004c2ccf8716c25e70a86f08" +dependencies = [ + "jsonptr", + "serde", + "serde_json", + "thiserror 1.0.69", +] + +[[package]] +name = "jsonptr" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5dea2b27dd239b2556ed7a25ba842fe47fd602e7fc7433c2a8d6106d4d9edd70" +dependencies = [ + "serde", + "serde_json", +] + +[[package]] +name = "keyboard-types" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b750dcadc39a09dbadd74e118f6dd6598df77fa01df0cfcdc52c28dece74528a" +dependencies = [ + "bitflags 2.11.0", + "serde", + "unicode-segmentation", +] + +[[package]] +name = "kuchikiki" +version = "0.8.8-speedreader" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02cb977175687f33fa4afa0c95c112b987ea1443e5a51c8f8ff27dc618270cc2" +dependencies = [ + "cssparser 0.29.6", + "html5ever 0.29.1", + "indexmap 2.14.0", + "selectors 0.24.0", +] + +[[package]] +name = "leb128fmt" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09edd9e8b54e49e587e4f6295a7d29c3ea94d469cb40ab8ca70b288248a81db2" + +[[package]] +name = "libappindicator" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "03589b9607c868cc7ae54c0b2a22c8dc03dd41692d48f2d7df73615c6a95dc0a" +dependencies = [ + "glib", + "gtk", + "gtk-sys", + "libappindicator-sys", + "log", +] + +[[package]] +name = "libappindicator-sys" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e9ec52138abedcc58dc17a7c6c0c00a2bdb4f3427c7f63fa97fd0d859155caf" +dependencies = [ + "gtk-sys", + "libloading", + "once_cell", +] + +[[package]] +name = "libc" +version = "0.2.184" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48f5d2a454e16a5ea0f4ced81bd44e4cfc7bd3a507b61887c99fd3538b28e4af" + +[[package]] +name = "libloading" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b67380fd3b2fbe7527a606e18729d21c6f3951633d0500574c4dc22d2d638b9f" +dependencies = [ + "cfg-if", + "winapi", +] + +[[package]] +name = "libredox" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e02f3bb43d335493c96bf3fd3a321600bf6bd07ed34bc64118e9293bdffea46c" +dependencies = [ + "libc", +] + +[[package]] +name = "libsqlite3-sys" +version = "0.28.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c10584274047cb335c23d3e61bcef8e323adae7c5c8c760540f73610177fc3f" +dependencies = [ + "cc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "linux-raw-sys" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32a66949e030da00e8c7d4434b251670a91556f4144941d37452769c25d58a53" + +[[package]] +name = "litemap" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92daf443525c4cce67b150400bc2316076100ce0b3686209eb8cf3c31612e6f0" + +[[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" + +[[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 0.8.9", + "string_cache_codegen 0.5.4", + "tendril 0.4.3", +] + +[[package]] +name = "markup5ever" +version = "0.38.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8983d30f2915feeaaab2d6babdd6bc7e9ed1a00b66b5e6d74df19aa9c0e91862" +dependencies = [ + "log", + "tendril 0.5.0", + "web_atoms", +] + +[[package]] +name = "match_token" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "88a9689d8d44bf9964484516275f5cd4c9b59457a6940c1d5d0ecbb94510a36b" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "matches" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2532096657941c2fea9c289d370a250971c689d4f143798ff67113ec042024a5" + +[[package]] +name = "memchr" +version = "2.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8ca58f447f06ed17d5fc4043ce1b10dd205e060fb3ce5b979b8ed8e59ff3f79" + +[[package]] +name = "memoffset" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "488016bfae457b036d996092f6cb448677611ce4449e970ceaf42695203f218a" +dependencies = [ + "autocfg", +] + +[[package]] +name = "mime" +version = "0.3.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" + +[[package]] +name = "miniz_oxide" +version = "0.8.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fa76a2c86f704bdb222d66965fb3d63269ce38518b83cb0575fca855ebb6316" +dependencies = [ + "adler2", + "simd-adler32", +] + +[[package]] +name = "mio" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50b7e5b27aa02a74bac8c3f23f448f8d87ff11f92d3aac1a6ed369ee08cc56c1" +dependencies = [ + "libc", + "wasi 0.11.1+wasi-snapshot-preview1", + "windows-sys 0.61.2", +] + +[[package]] +name = "muda" +version = "0.17.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c9fec5a4e89860383d778d10563a605838f8f0b2f9303868937e5ff32e86177" +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.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "465500e14ea162429d264d44189adc38b199b62b1c21eea9f69e4b73cb03bbf2" +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.11.0", + "jni-sys 0.3.1", + "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 0.3.1", +] + +[[package]] +name = "new_debug_unreachable" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "650eef8c711430f1a879fdd01d4745a7deea475becfb90269c06775983bbf086" + +[[package]] +name = "nodrop" +version = "0.1.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72ef4a56884ca558e5ddb05a1d1e7e1bfd9a68d9ed024c21704cc98872dae1bb" + +[[package]] +name = "num-conv" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6673768db2d862beb9b39a78fdcb1a69439615d5794a1be50caa9bc92c81967" + +[[package]] +name = "num-traits" +version = "0.2.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +dependencies = [ + "autocfg", +] + +[[package]] +name = "num_enum" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d0bca838442ec211fa11de3a8b0e0e8f3a4522575b5c4c06ed722e005036f26" +dependencies = [ + "num_enum_derive", + "rustversion", +] + +[[package]] +name = "num_enum_derive" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "680998035259dcfcafe653688bf2aa6d3e2dc05e98be6ab46afb089dc84f1df8" +dependencies = [ + "proc-macro-crate 3.5.0", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "objc2" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3a12a8ed07aefc768292f076dc3ac8c48f3781c8f2d5851dd3d98950e8c5a89f" +dependencies = [ + "objc2-encode", + "objc2-exception-helper", +] + +[[package]] +name = "objc2-app-kit" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d49e936b501e5c5bf01fda3a9452ff86dc3ea98ad5f283e1455153142d97518c" +dependencies = [ + "bitflags 2.11.0", + "block2", + "objc2", + "objc2-core-foundation", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-foundation" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2a180dd8642fa45cdb7dd721cd4c11b1cadd4929ce112ebd8b9f5803cc79d536" +dependencies = [ + "bitflags 2.11.0", + "dispatch2", + "objc2", +] + +[[package]] +name = "objc2-core-graphics" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e022c9d066895efa1345f8e33e584b9f958da2fd4cd116792e15e07e4720a807" +dependencies = [ + "bitflags 2.11.0", + "dispatch2", + "objc2", + "objc2-core-foundation", + "objc2-io-surface", +] + +[[package]] +name = "objc2-encode" +version = "4.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef25abbcd74fb2609453eb695bd2f860d389e457f67dc17cafc8b8cbc89d0c33" + +[[package]] +name = "objc2-exception-helper" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7a1c5fbb72d7735b076bb47b578523aedc40f3c439bea6dfd595c089d79d98a" +dependencies = [ + "cc", +] + +[[package]] +name = "objc2-foundation" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3e0adef53c21f888deb4fa59fc59f7eb17404926ee8a6f59f5df0fd7f9f3272" +dependencies = [ + "bitflags 2.11.0", + "block2", + "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.11.0", + "objc2", + "objc2-core-foundation", +] + +[[package]] +name = "objc2-quartz-core" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96c1358452b371bf9f104e21ec536d37a650eb10f7ee379fff67d2e08d537f1f" +dependencies = [ + "bitflags 2.11.0", + "objc2", + "objc2-core-foundation", + "objc2-foundation", +] + +[[package]] +name = "objc2-ui-kit" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d87d638e33c06f577498cbcc50491496a3ed4246998a7fbba7ccb98b1e7eab22" +dependencies = [ + "bitflags 2.11.0", + "objc2", + "objc2-core-foundation", + "objc2-foundation", +] + +[[package]] +name = "objc2-web-kit" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2e5aaab980c433cf470df9d7af96a7b46a9d892d521a2cbbb2f8a4c16751e7f" +dependencies = [ + "bitflags 2.11.0", + "block2", + "objc2", + "objc2-app-kit", + "objc2-core-foundation", + "objc2-foundation", +] + +[[package]] +name = "once_cell" +version = "1.21.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f7c3e4beb33f85d45ae3e3a1792185706c8e16d043238c593331cc7cd313b50" + +[[package]] +name = "opaque-debug" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c08d65885ee38876c4f86fa503fb49d7b507c2b62552df7c70b2fce627e06381" + +[[package]] +name = "openssl" +version = "0.10.77" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfe4646e360ec77dff7dde40ed3d6c5fee52d156ef4a62f53973d38294dad87f" +dependencies = [ + "bitflags 2.11.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.117", +] + +[[package]] +name = "openssl-probe" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c87def4c32ab89d880effc9e097653c8da5d6ef28e6b539d313baaacfbafcbe" + +[[package]] +name = "openssl-sys" +version = "0.9.113" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ad2f2c0eba47118757e4c6d2bff2838f3e0523380021356e7875e858372ce644" +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 = "orchai" +version = "0.1.0" +dependencies = [ + "aes-gcm", + "base64 0.22.1", + "chrono", + "dirs 5.0.1", + "rand 0.8.5", + "reqwest 0.12.28", + "rusqlite", + "serde", + "serde_json", + "tauri", + "tauri-build", + "tauri-plugin-dialog", + "tokio", + "uuid", +] + +[[package]] +name = "pango" +version = "0.18.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ca27ec1eb0457ab26f3036ea52229edbdb74dee1edd29063f5b9b010e7ebee4" +dependencies = [ + "gio", + "glib", + "libc", + "once_cell", + "pango-sys", +] + +[[package]] +name = "pango-sys" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "436737e391a843e5933d6d9aa102cb126d501e815b83601365a948a518555dc5" +dependencies = [ + "glib-sys", + "gobject-sys", + "libc", + "system-deps", +] + +[[package]] +name = "parking_lot" +version = "0.12.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93857453250e3077bd71ff98b6a65ea6621a19bb0f559a85248955ac12c45a1a" +dependencies = [ + "lock_api", + "parking_lot_core", +] + +[[package]] +name = "parking_lot_core" +version = "0.9.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2621685985a2ebf1c516881c026032ac7deafcda1a2c9b7850dc81e3dfcb64c1" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall", + "smallvec", + "windows-link 0.2.1", +] + +[[package]] +name = "percent-encoding" +version = "2.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b4f627cb1b25917193a259e49bdad08f671f8d9708acfd5fe0a8c1455d87220" + +[[package]] +name = "phf" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3dfb61232e34fcb633f43d12c58f83c1df82962dcdfa565a4e866ffc17dafe12" +dependencies = [ + "phf_shared 0.8.0", +] + +[[package]] +name = "phf" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fabbf1ead8a5bcbc20f5f8b939ee3f5b0f6f281b6ad3468b84656b658b455259" +dependencies = [ + "phf_macros 0.10.0", + "phf_shared 0.10.0", + "proc-macro-hack", +] + +[[package]] +name = "phf" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd6780a80ae0c52cc120a26a1a42c1ae51b247a253e4e06113d23d2c2edd078" +dependencies = [ + "phf_macros 0.11.3", + "phf_shared 0.11.3", +] + +[[package]] +name = "phf" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c1562dc717473dbaa4c1f85a36410e03c047b2e7df7f45ee938fbef64ae7fadf" +dependencies = [ + "phf_macros 0.13.1", + "phf_shared 0.13.1", + "serde", +] + +[[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_codegen" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49aa7f9d80421bca176ca8dbfebe668cc7a2684708594ec9f3c0db0805d5d6e1" +dependencies = [ + "phf_generator 0.13.1", + "phf_shared 0.13.1", +] + +[[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_generator" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "135ace3a761e564ec88c03a77317a7c6b80bb7f7135ef2544dbe054243b89737" +dependencies = [ + "fastrand", + "phf_shared 0.13.1", +] + +[[package]] +name = "phf_macros" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "58fdf3184dd560f160dd73922bea2d5cd6e8f064bf4b13110abd81b03697b4e0" +dependencies = [ + "phf_generator 0.10.0", + "phf_shared 0.10.0", + "proc-macro-hack", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "phf_macros" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f84ac04429c13a7ff43785d75ad27569f2951ce0ffd30a3321230db2fc727216" +dependencies = [ + "phf_generator 0.11.3", + "phf_shared 0.11.3", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "phf_macros" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "812f032b54b1e759ccd5f8b6677695d5268c588701effba24601f6932f8269ef" +dependencies = [ + "phf_generator 0.13.1", + "phf_shared 0.13.1", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "phf_shared" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c00cf8b9eafe68dde5e9eaa2cef8ee84a9336a47d566ec55ca16589633b65af7" +dependencies = [ + "siphasher 0.3.11", +] + +[[package]] +name = "phf_shared" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6796ad771acdc0123d2a88dc428b5e38ef24456743ddb1744ed628f9815c096" +dependencies = [ + "siphasher 0.3.11", +] + +[[package]] +name = "phf_shared" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67eabc2ef2a60eb7faa00097bd1ffdb5bd28e62bf39990626a582201b7a754e5" +dependencies = [ + "siphasher 1.0.2", +] + +[[package]] +name = "phf_shared" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e57fef6bc5981e38c2ce2d63bfa546861309f875b8a75f092d1d54ae2d64f266" +dependencies = [ + "siphasher 1.0.2", +] + +[[package]] +name = "pin-project-lite" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a89322df9ebe1c1578d689c92318e070967d1042b512afbe49518723f4e6d5cd" + +[[package]] +name = "pkg-config" +version = "0.3.33" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19f132c84eca552bf34cab8ec81f1c1dcc229b811638f9d283dceabe58c5569e" + +[[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.14.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 = "polyval" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d1fe60d06143b2430aa532c94cfe9e29783047f06c0d7fd359a9a51b729fa25" +dependencies = [ + "cfg-if", + "cpufeatures", + "opaque-debug", + "universal-hash", +] + +[[package]] +name = "potential_utf" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0103b1cef7ec0cf76490e969665504990193874ea05c85ff9bab8b911d0a0564" +dependencies = [ + "zerovec", +] + +[[package]] +name = "powerfmt" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391" + +[[package]] +name = "ppv-lite86" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85eae3c4ed2f50dcfe72643da4befc30deadb458a9b590d720cde2f2b1e97da9" +dependencies = [ + "zerocopy", +] + +[[package]] +name = "precomputed-hash" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "925383efa346730478fb4838dbe9137d2a47675ad789c546d150a6e1dd4ab31c" + +[[package]] +name = "prettyplease" +version = "0.2.37" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "479ca8adacdd7ce8f1fb39ce9ecccbfe93a3f1344b3d0d97f20bc0196208f62b" +dependencies = [ + "proc-macro2", + "syn 2.0.117", +] + +[[package]] +name = "proc-macro-crate" +version = "1.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f4c021e1093a56626774e81216a4ce732a735e5bad4868a03f3ed65ca0c3919" +dependencies = [ + "once_cell", + "toml_edit 0.19.15", +] + +[[package]] +name = "proc-macro-crate" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b00f26d3400549137f92511a46ac1cd8ce37cb5598a96d382381458b992a5d24" +dependencies = [ + "toml_datetime 0.6.3", + "toml_edit 0.20.2", +] + +[[package]] +name = "proc-macro-crate" +version = "3.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e67ba7e9b2b56446f1d419b1d807906278ffa1a658a8a5d8a39dcb1f5a78614f" +dependencies = [ + "toml_edit 0.25.11+spec-1.1.0", +] + +[[package]] +name = "proc-macro-error" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c" +dependencies = [ + "proc-macro-error-attr", + "proc-macro2", + "quote", + "syn 1.0.109", + "version_check", +] + +[[package]] +name = "proc-macro-error-attr" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869" +dependencies = [ + "proc-macro2", + "quote", + "version_check", +] + +[[package]] +name = "proc-macro-hack" +version = "0.5.20+deprecated" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc375e1527247fe1a97d8b7156678dfe7c1af2fc075c9a4db3690ecd2a148068" + +[[package]] +name = "proc-macro2" +version = "1.0.106" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fd00f0bb2e90d81d1044c2b32617f68fcb9fa3bb7640c23e9c748e53fb30934" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "quick-xml" +version = "0.38.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b66c2058c55a409d601666cffe35f04333cf1013010882cec174a7467cd4e21c" +dependencies = [ + "memchr", +] + +[[package]] +name = "quote" +version = "1.0.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41f2619966050689382d2b44f664f4bc593e129785a36d6ee376ddf37259b924" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "r-efi" +version = "5.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69cdb34c158ceb288df11e18b4bd39de994f6657d83847bdffdbd7f346754b0f" + +[[package]] +name = "r-efi" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8dcc9c7d52a811697d2151c701e0d08956f92b0e24136cf4cf27b57a6a0d9bf" + +[[package]] +name = "rand" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03" +dependencies = [ + "getrandom 0.1.16", + "libc", + "rand_chacha 0.2.2", + "rand_core 0.5.1", + "rand_hc", + "rand_pcg", +] + +[[package]] +name = "rand" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" +dependencies = [ + "libc", + "rand_chacha 0.3.1", + "rand_core 0.6.4", +] + +[[package]] +name = "rand_chacha" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402" +dependencies = [ + "ppv-lite86", + "rand_core 0.5.1", +] + +[[package]] +name = "rand_chacha" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" +dependencies = [ + "ppv-lite86", + "rand_core 0.6.4", +] + +[[package]] +name = "rand_core" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19" +dependencies = [ + "getrandom 0.1.16", +] + +[[package]] +name = "rand_core" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" +dependencies = [ + "getrandom 0.2.17", +] + +[[package]] +name = "rand_hc" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c" +dependencies = [ + "rand_core 0.5.1", +] + +[[package]] +name = "rand_pcg" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "16abd0c1b639e9eb4d7c50c0b8100b0d0f849be2349829c740fe8e6eb4816429" +dependencies = [ + "rand_core 0.5.1", +] + +[[package]] +name = "raw-window-handle" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20675572f6f24e9e76ef639bc5552774ed45f1c30e2951e1e99c59888861c539" + +[[package]] +name = "redox_syscall" +version = "0.5.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed2bf2547551a7053d6fdfafda3f938979645c44812fbfcda098faae3f1a362d" +dependencies = [ + "bitflags 2.11.0", +] + +[[package]] +name = "redox_users" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba009ff324d1fc1b900bd1fdb31564febe58a8ccc8a6fdbb93b543d33b13ca43" +dependencies = [ + "getrandom 0.2.17", + "libredox", + "thiserror 1.0.69", +] + +[[package]] +name = "redox_users" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4e608c6638b9c18977b00b475ac1f28d14e84b27d8d42f70e0bf1e3dec127ac" +dependencies = [ + "getrandom 0.2.17", + "libredox", + "thiserror 2.0.18", +] + +[[package]] +name = "ref-cast" +version = "1.0.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f354300ae66f76f1c85c5f84693f0ce81d747e2c3f21a45fef496d89c960bf7d" +dependencies = [ + "ref-cast-impl", +] + +[[package]] +name = "ref-cast-impl" +version = "1.0.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7186006dcb21920990093f30e3dea63b7d6e977bf1256be20c3563a5db070da" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "regex" +version = "1.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e10754a14b9137dd7b1e3e5b0493cc9171fdd105e0ab477f51b72e7f3ac0e276" +dependencies = [ + "aho-corasick", + "memchr", + "regex-automata", + "regex-syntax", +] + +[[package]] +name = "regex-automata" +version = "0.4.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e1dd4122fc1595e8162618945476892eefca7b88c52820e74af6262213cae8f" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", +] + +[[package]] +name = "regex-syntax" +version = "0.8.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc897dd8d9e8bd1ed8cdad82b5966c3e0ecae09fb1907d58efaa013543185d0a" + +[[package]] +name = "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", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "reqwest" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ab3f43e3283ab1488b624b44b0e988d0acea0b3214e694730a055cb6b2efa801" +dependencies = [ + "base64 0.22.1", + "bytes", + "futures-core", + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-util", + "js-sys", + "log", + "percent-encoding", + "pin-project-lite", + "serde", + "serde_json", + "sync_wrapper", + "tokio", + "tokio-util", + "tower", + "tower-http", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-streams", + "web-sys", +] + +[[package]] +name = "rfd" +version = "0.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a15ad77d9e70a92437d8f74c35d99b4e4691128df018833e99f90bcd36152672" +dependencies = [ + "block2", + "dispatch2", + "glib-sys", + "gobject-sys", + "gtk-sys", + "js-sys", + "log", + "objc2", + "objc2-app-kit", + "objc2-core-foundation", + "objc2-foundation", + "raw-window-handle", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "windows-sys 0.60.2", +] + +[[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 = "rusqlite" +version = "0.31.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b838eba278d213a8beaf485bd313fd580ca4505a00d5871caeb1457c55322cae" +dependencies = [ + "bitflags 2.11.0", + "fallible-iterator", + "fallible-streaming-iterator", + "hashlink", + "libsqlite3-sys", + "smallvec", +] + +[[package]] +name = "rustc-hash" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94300abf3f1ae2e2b8ffb7b58043de3d399c73fa6f4b73826402a5c457614dbe" + +[[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.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6fe4565b9518b83ef4f91bb47ce29620ca828bd32cb7e408f0062e9930ba190" +dependencies = [ + "bitflags 2.11.0", + "errno", + "libc", + "linux-raw-sys", + "windows-sys 0.61.2", +] + +[[package]] +name = "rustls" +version = "0.23.38" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69f9466fb2c14ea04357e91413efb882e2a6d4a406e625449bc0a5d360d53a21" +dependencies = [ + "once_cell", + "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.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20a6af516fea4b20eccceaf166e8aa666ac996208e8a644ce3ef5aa783bc7cd4" +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.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9774ba4a74de5f7b1c1451ed6cd5285a32eddb5cccb8cc655a4e50009e06477f" + +[[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.29" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91c1b7e4904c873ef0710c1f407dde2e6287de2bebc1bbbf7d430bb7cbffd939" +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.117", +] + +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + +[[package]] +name = "security-framework" +version = "3.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7f4bc775c73d9a02cde8bf7b2ec4c9d12743edf609006c7facc23998404cd1d" +dependencies = [ + "bitflags 2.11.0", + "core-foundation 0.10.1", + "core-foundation-sys", + "libc", + "security-framework-sys", +] + +[[package]] +name = "security-framework-sys" +version = "2.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ce2691df843ecc5d231c0b14ece2acc3efb62c0a398c7e1d875f3983ce020e3" +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 0.29.6", + "derive_more 0.99.20", + "fxhash", + "log", + "phf 0.8.0", + "phf_codegen 0.8.0", + "precomputed-hash", + "servo_arc 0.2.0", + "smallvec", +] + +[[package]] +name = "selectors" +version = "0.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c5d9c0c92a92d33f08817311cf3f2c29a3538a8240e94a6a3c622ce652d7e00c" +dependencies = [ + "bitflags 2.11.0", + "cssparser 0.36.0", + "derive_more 2.1.1", + "log", + "new_debug_unreachable", + "phf 0.13.1", + "phf_codegen 0.13.1", + "precomputed-hash", + "rustc-hash", + "servo_arc 0.4.3", + "smallvec", +] + +[[package]] +name = "semver" +version = "1.0.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a7852d02fc848982e0c167ef163aaff9cd91dc640ba85e263cb1ce46fae51cd" +dependencies = [ + "serde", + "serde_core", +] + +[[package]] +name = "serde" +version = "1.0.228" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a8e94ea7f378bd32cbbd37198a4a91436180c5bb472411e48b5ec2e2124ae9e" +dependencies = [ + "serde_core", + "serde_derive", +] + +[[package]] +name = "serde-untagged" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f9faf48a4a2d2693be24c6289dbe26552776eb7737074e6722891fadbe6c5058" +dependencies = [ + "erased-serde", + "serde", + "serde_core", + "typeid", +] + +[[package]] +name = "serde_core" +version = "1.0.228" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41d385c7d4ca58e59fc732af25c3983b67ac852c1a25000afe1175de458b67ad" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde_derive" +version = "1.0.228" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d540f220d3187173da220f885ab66608367b6574e925011a9353e4badda91d79" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "serde_derive_internals" +version = "0.29.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "18d26a20a969b9e3fdf2fc2d9f21eda6c40e2de84c9408bb5d3b05d499aae711" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "serde_json" +version = "1.0.149" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "83fc039473c5595ace860d8c4fafa220ff474b3fc6bfdb4293327f1a37e94d86" +dependencies = [ + "itoa", + "memchr", + "serde", + "serde_core", + "zmij", +] + +[[package]] +name = "serde_repr" +version = "0.1.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "175ee3e80ae9982737ca543e96133087cbd9a485eecc3bc4de9c1a37b47ea59c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "serde_spanned" +version = "0.6.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf41e0cfaf7226dca15e8197172c295a782857fcb97fad1808a166870dee75a3" +dependencies = [ + "serde", +] + +[[package]] +name = "serde_spanned" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6662b5879511e06e8999a8a235d848113e942c9124f211511b16466ee2995f26" +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.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd5414fad8e6907dbdd5bc441a50ae8d6e26151a03b1de04d89a5576de61d01f" +dependencies = [ + "base64 0.22.1", + "chrono", + "hex", + "indexmap 1.9.3", + "indexmap 2.14.0", + "schemars 0.9.0", + "schemars 1.2.1", + "serde_core", + "serde_json", + "serde_with_macros", + "time", +] + +[[package]] +name = "serde_with_macros" +version = "3.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3db8978e608f1fe7357e211969fd9abdcae80bac1ba7a3369bb7eb6b404eb65" +dependencies = [ + "darling", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "serialize-to-javascript" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "04f3666a07a197cdb77cdf306c32be9b7f598d7060d50cfd4d5aa04bfd92f6c5" +dependencies = [ + "serde", + "serde_json", + "serialize-to-javascript-impl", +] + +[[package]] +name = "serialize-to-javascript-impl" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "772ee033c0916d670af7860b6e1ef7d658a4629a6d0b4c8c3e67f09b3765b75d" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "servo_arc" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d52aa42f8fdf0fed91e5ce7f23d8138441002fa31dca008acf47e6fd4721f741" +dependencies = [ + "nodrop", + "stable_deref_trait", +] + +[[package]] +name = "servo_arc" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "170fb83ab34de17dc69aa7c67482b22218ddb85da56546f9bd6b929e32a05930" +dependencies = [ + "stable_deref_trait", +] + +[[package]] +name = "sha2" +version = "0.10.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7507d819769d01a365ab707794a4084392c824f54a7a6a7862f8c3d0892b283" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[package]] +name = "shlex" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" + +[[package]] +name = "simd-adler32" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "703d5c7ef118737c72f1af64ad2f6f8c5e1921f818cdcb97b8fe6fc69bf66214" + +[[package]] +name = "siphasher" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38b58827f4464d87d377d175e90bf58eb00fd8716ff0a62f80356b5e61555d0d" + +[[package]] +name = "siphasher" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2aa850e253778c88a04c3d7323b043aeda9d3e30d5971937c1855769763678e" + +[[package]] +name = "slab" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c790de23124f9ab44544d7ac05d60440adc586479ce501c1d6d7da3cd8c9cf5" + +[[package]] +name = "smallvec" +version = "1.15.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67b1b7a3b5fe4f1376887184045fcf45c69e92af734b7aaddc05fb777b6fbd03" + +[[package]] +name = "socket2" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3a766e1110788c36f4fa1c2b71b387a7815aa65f88ce0229841826633d93723e" +dependencies = [ + "libc", + "windows-sys 0.61.2", +] + +[[package]] +name = "softbuffer" +version = "0.4.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aac18da81ebbf05109ab275b157c22a653bb3c12cf884450179942f81bcbf6c3" +dependencies = [ + "bytemuck", + "js-sys", + "ndk", + "objc2", + "objc2-core-foundation", + "objc2-core-graphics", + "objc2-foundation", + "objc2-quartz-core", + "raw-window-handle", + "redox_syscall", + "tracing", + "wasm-bindgen", + "web-sys", + "windows-sys 0.61.2", +] + +[[package]] +name = "soup3" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "471f924a40f31251afc77450e781cb26d55c0b650842efafc9c6cbd2f7cc4f9f" +dependencies = [ + "futures-channel", + "gio", + "glib", + "libc", + "soup3-sys", +] + +[[package]] +name = "soup3-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ebe8950a680a12f24f15ebe1bf70db7af98ad242d9db43596ad3108aab86c27" +dependencies = [ + "gio-sys", + "glib-sys", + "gobject-sys", + "libc", + "system-deps", +] + +[[package]] +name = "stable_deref_trait" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ce2be8dc25455e1f91df71bfa12ad37d7af1092ae736f3a6cd0e37bc7810596" + +[[package]] +name = "string_cache" +version = "0.8.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf776ba3fa74f83bf4b63c3dcbbf82173db2632ed8452cb2d891d33f459de70f" +dependencies = [ + "new_debug_unreachable", + "parking_lot", + "phf_shared 0.11.3", + "precomputed-hash", + "serde", +] + +[[package]] +name = "string_cache" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a18596f8c785a729f2819c0f6a7eae6ebeebdfffbfe4214ae6b087f690e31901" +dependencies = [ + "new_debug_unreachable", + "parking_lot", + "phf_shared 0.13.1", + "precomputed-hash", +] + +[[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 = "string_cache_codegen" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "585635e46db231059f76c5849798146164652513eb9e8ab2685939dd90f29b69" +dependencies = [ + "phf_generator 0.13.1", + "phf_shared 0.13.1", + "proc-macro2", + "quote", +] + +[[package]] +name = "strsim" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" + +[[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.117" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e665b8803e7b1d2a727f4023456bbbbe74da67099c585258af0ad9c5013b9b99" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "sync_wrapper" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bf256ce5efdfa370213c1dabab5935a12e49f2c58d15e9eac2870d3b4f27263" +dependencies = [ + "futures-core", +] + +[[package]] +name = "synstructure" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "728a70f3dbaf5bab7f0c4b1ac8d7ae5ea60a4b5549c8a5914361c99147a709d2" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "system-configuration" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a13f3d0daba03132c0aa9767f98351b3488edc2c100cda2d2ec2b04f3d8d3c8b" +dependencies = [ + "bitflags 2.11.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.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9103edf55f2da3c82aea4c7fab7c4241032bfeea0e71fa557d98e00e7ce7cc20" +dependencies = [ + "bitflags 2.11.0", + "block2", + "core-foundation 0.10.1", + "core-graphics", + "crossbeam-channel", + "dispatch2", + "dlopen2", + "dpi", + "gdkwayland-sys", + "gdkx11-sys", + "gtk", + "jni", + "libc", + "log", + "ndk", + "ndk-context", + "ndk-sys", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "once_cell", + "parking_lot", + "raw-window-handle", + "tao-macros", + "unicode-segmentation", + "url", + "windows", + "windows-core 0.61.2", + "windows-version", + "x11-dl", +] + +[[package]] +name = "tao-macros" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4e16beb8b2ac17db28eab8bca40e62dbfbb34c0fcdc6d9826b11b7b5d047dfd" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "target-lexicon" +version = "0.12.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61c41af27dd6d1e27b1b16b489db798443478cef1f06a660c96db617ba5de3b1" + +[[package]] +name = "tauri" +version = "2.10.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da77cc00fb9028caf5b5d4650f75e31f1ef3693459dfca7f7e506d1ecef0ba2d" +dependencies = [ + "anyhow", + "bytes", + "cookie", + "dirs 6.0.0", + "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.2", + "serde", + "serde_json", + "serde_repr", + "serialize-to-javascript", + "swift-rs", + "tauri-build", + "tauri-macros", + "tauri-runtime", + "tauri-runtime-wry", + "tauri-utils", + "thiserror 2.0.18", + "tokio", + "tray-icon", + "url", + "webkit2gtk", + "webview2-com", + "window-vibrancy", + "windows", +] + +[[package]] +name = "tauri-build" +version = "2.5.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4bbc990d1dbf57a8e1c7fa2327f2a614d8b757805603c1b9ba5c81bade09fd4d" +dependencies = [ + "anyhow", + "cargo_toml", + "dirs 6.0.0", + "glob", + "heck 0.5.0", + "json-patch", + "schemars 0.8.22", + "semver", + "serde", + "serde_json", + "tauri-utils", + "tauri-winres", + "toml 0.9.12+spec-1.1.0", + "walkdir", +] + +[[package]] +name = "tauri-codegen" +version = "2.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d4a24476afd977c5d5d169f72425868613d82747916dd29e0a357c84c4bd6d29" +dependencies = [ + "base64 0.22.1", + "brotli", + "ico", + "json-patch", + "plist", + "png", + "proc-macro2", + "quote", + "semver", + "serde", + "serde_json", + "sha2", + "syn 2.0.117", + "tauri-utils", + "thiserror 2.0.18", + "time", + "url", + "uuid", + "walkdir", +] + +[[package]] +name = "tauri-macros" +version = "2.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d39b349a98dadaffebb73f0a40dcd1f23c999211e5a2e744403db384d0c33de7" +dependencies = [ + "heck 0.5.0", + "proc-macro2", + "quote", + "syn 2.0.117", + "tauri-codegen", + "tauri-utils", +] + +[[package]] +name = "tauri-plugin" +version = "2.5.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ddde7d51c907b940fb573006cdda9a642d6a7c8153657e88f8a5c3c9290cd4aa" +dependencies = [ + "anyhow", + "glob", + "plist", + "schemars 0.8.22", + "serde", + "serde_json", + "tauri-utils", + "toml 0.9.12+spec-1.1.0", + "walkdir", +] + +[[package]] +name = "tauri-plugin-dialog" +version = "2.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1fa4150c95ae391946cc8b8f905ab14797427caba3a8a2f79628e956da91809" +dependencies = [ + "log", + "raw-window-handle", + "rfd", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "tauri-plugin-fs", + "thiserror 2.0.18", + "url", +] + +[[package]] +name = "tauri-plugin-fs" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "36e1ec28b79f3d0683f4507e1615c36292c0ea6716668770d4396b9b39871ed8" +dependencies = [ + "anyhow", + "dunce", + "glob", + "log", + "objc2-foundation", + "percent-encoding", + "schemars 0.8.22", + "serde", + "serde_json", + "serde_repr", + "tauri", + "tauri-plugin", + "tauri-utils", + "thiserror 2.0.18", + "toml 0.9.12+spec-1.1.0", + "url", +] + +[[package]] +name = "tauri-runtime" +version = "2.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2826d79a3297ed08cd6ea7f412644ef58e32969504bc4fbd8d7dbeabc4445ea2" +dependencies = [ + "cookie", + "dpi", + "gtk", + "http", + "jni", + "objc2", + "objc2-ui-kit", + "objc2-web-kit", + "raw-window-handle", + "serde", + "serde_json", + "tauri-utils", + "thiserror 2.0.18", + "url", + "webkit2gtk", + "webview2-com", + "windows", +] + +[[package]] +name = "tauri-runtime-wry" +version = "2.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e11ea2e6f801d275fdd890d6c9603736012742a1c33b96d0db788c9cdebf7f9e" +dependencies = [ + "gtk", + "http", + "jni", + "log", + "objc2", + "objc2-app-kit", + "once_cell", + "percent-encoding", + "raw-window-handle", + "softbuffer", + "tao", + "tauri-runtime", + "tauri-utils", + "url", + "webkit2gtk", + "webview2-com", + "windows", + "wry", +] + +[[package]] +name = "tauri-utils" +version = "2.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "219a1f983a2af3653f75b5747f76733b0da7ff03069c7a41901a5eb3ace4557d" +dependencies = [ + "anyhow", + "brotli", + "cargo_metadata", + "ctor", + "dunce", + "glob", + "html5ever 0.29.1", + "http", + "infer", + "json-patch", + "kuchikiki", + "log", + "memchr", + "phf 0.11.3", + "proc-macro2", + "quote", + "regex", + "schemars 0.8.22", + "semver", + "serde", + "serde-untagged", + "serde_json", + "serde_with", + "swift-rs", + "thiserror 2.0.18", + "toml 0.9.12+spec-1.1.0", + "url", + "urlpattern", + "uuid", + "walkdir", +] + +[[package]] +name = "tauri-winres" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1087b111fe2b005e42dbdc1990fc18593234238d47453b0c99b7de1c9ab2c1e0" +dependencies = [ + "dunce", + "embed-resource", + "toml 0.9.12+spec-1.1.0", +] + +[[package]] +name = "tempfile" +version = "3.27.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32497e9a4c7b38532efcdebeef879707aa9f794296a4f0244f6f69e9bc8574bd" +dependencies = [ + "fastrand", + "getrandom 0.4.2", + "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 = "tendril" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4790fc369d5a530f4b544b094e31388b9b3a37c0f4652ade4505945f5660d24" +dependencies = [ + "new_debug_unreachable", + "utf-8", +] + +[[package]] +name = "thiserror" +version = "1.0.69" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6aaf5339b578ea85b50e080feb250a3e8ae8cfcdff9a461c9ec2904bc923f52" +dependencies = [ + "thiserror-impl 1.0.69", +] + +[[package]] +name = "thiserror" +version = "2.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4288b5bcbc7920c07a1149a35cf9590a2aa808e0bc1eafaade0b80947865fbc4" +dependencies = [ + "thiserror-impl 2.0.18", +] + +[[package]] +name = "thiserror-impl" +version = "1.0.69" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "thiserror-impl" +version = "2.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebc4ee7f67670e9b64d05fa4253e753e016c6c95ff35b89b7941d6b856dec1d5" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "time" +version = "0.3.47" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "743bd48c283afc0388f9b8827b976905fb217ad9e647fae3a379a9283c4def2c" +dependencies = [ + "deranged", + "itoa", + "num-conv", + "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.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8323304221c2a851516f22236c5722a72eaa19749016521d6dff0824447d96d" +dependencies = [ + "displaydoc", + "zerovec", +] + +[[package]] +name = "tokio" +version = "1.51.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f66bf9585cda4b724d3e78ab34b73fb2bbaba9011b9bfdf69dc836382ea13b8c" +dependencies = [ + "bytes", + "libc", + "mio", + "pin-project-lite", + "socket2", + "tokio-macros", + "windows-sys 0.61.2", +] + +[[package]] +name = "tokio-macros" +version = "2.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "385a6cb71ab9ab790c5fe8d67f1645e6c450a7ce006a33de03daa956cf70a496" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[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-util" +version = "0.7.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ae9cec805b01e8fc3fd2fe289f89149a9b66dd16786abd8b19cfa7b48cb0098" +dependencies = [ + "bytes", + "futures-core", + "futures-sink", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "toml" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "185d8ab0dfbb35cf1399a6344d8484209c088f75f8f68230da55d48d95d43e3d" +dependencies = [ + "serde", + "serde_spanned 0.6.9", + "toml_datetime 0.6.3", + "toml_edit 0.20.2", +] + +[[package]] +name = "toml" +version = "0.9.12+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf92845e79fc2e2def6a5d828f0801e29a2f8acc037becc5ab08595c7d5e9863" +dependencies = [ + "indexmap 2.14.0", + "serde_core", + "serde_spanned 1.1.1", + "toml_datetime 0.7.5+spec-1.1.0", + "toml_parser", + "toml_writer", + "winnow 0.7.15", +] + +[[package]] +name = "toml_datetime" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7cda73e2f1397b1262d6dfdcef8aafae14d1de7748d66822d3bfeeb6d03e5e4b" +dependencies = [ + "serde", +] + +[[package]] +name = "toml_datetime" +version = "0.7.5+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92e1cfed4a3038bc5a127e35a2d360f145e1f4b971b551a2ba5fd7aedf7e1347" +dependencies = [ + "serde_core", +] + +[[package]] +name = "toml_datetime" +version = "1.1.1+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3165f65f62e28e0115a00b2ebdd37eb6f3b641855f9d636d3cd4103767159ad7" +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.14.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.14.0", + "serde", + "serde_spanned 0.6.9", + "toml_datetime 0.6.3", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.25.11+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b59c4d22ed448339746c59b905d24568fcbb3ab65a500494f7b8c3e97739f2b" +dependencies = [ + "indexmap 2.14.0", + "toml_datetime 1.1.1+spec-1.1.0", + "toml_parser", + "winnow 1.0.1", +] + +[[package]] +name = "toml_parser" +version = "1.1.2+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2abe9b86193656635d2411dc43050282ca48aa31c2451210f4202550afb7526" +dependencies = [ + "winnow 1.0.1", +] + +[[package]] +name = "toml_writer" +version = "1.1.1+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "756daf9b1013ebe47a8776667b466417e2d4c5679d441c26230efd9ef78692db" + +[[package]] +name = "tower" +version = "0.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebe5ef63511595f1344e2d5cfa636d973292adc0eec1f0ad45fae9f0851ab1d4" +dependencies = [ + "futures-core", + "futures-util", + "pin-project-lite", + "sync_wrapper", + "tokio", + "tower-layer", + "tower-service", +] + +[[package]] +name = "tower-http" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d4e6559d53cc268e5031cd8429d05415bc4cb4aefc4aa5d6cc35fbf5b924a1f8" +dependencies = [ + "bitflags 2.11.0", + "bytes", + "futures-util", + "http", + "http-body", + "iri-string", + "pin-project-lite", + "tower", + "tower-layer", + "tower-service", +] + +[[package]] +name = "tower-layer" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "121c2a6cda46980bb0fcd1647ffaf6cd3fc79a013de288782836f6df9c48780e" + +[[package]] +name = "tower-service" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8df9b6e13f2d32c91b9bd719c00d1958837bc7dec474d94952798cc8e69eeec3" + +[[package]] +name = "tracing" +version = "0.1.44" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63e71662fa4b2a2c3a26f570f037eb95bb1f85397f3cd8076caed2f026a6d100" +dependencies = [ + "pin-project-lite", + "tracing-core", +] + +[[package]] +name = "tracing-core" +version = "0.1.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db97caf9d906fbde555dd62fa95ddba9eecfd14cb388e4f491a66d74cd5fb79a" +dependencies = [ + "once_cell", +] + +[[package]] +name = "tray-icon" +version = "0.21.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a5e85aa143ceb072062fc4d6356c1b520a51d636e7bc8e77ec94be3608e5e80c" +dependencies = [ + "crossbeam-channel", + "dirs 6.0.0", + "libappindicator", + "muda", + "objc2", + "objc2-app-kit", + "objc2-core-foundation", + "objc2-core-graphics", + "objc2-foundation", + "once_cell", + "png", + "serde", + "thiserror 2.0.18", + "windows-sys 0.60.2", +] + +[[package]] +name = "try-lock" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" + +[[package]] +name = "typeid" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bc7d623258602320d5c55d1bc22793b57daff0ec7efc270ea7d55ce1d5f5471c" + +[[package]] +name = "typenum" +version = "1.19.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "562d481066bde0658276a35467c4af00bdc6ee726305698a55b86e61d7ad82bb" + +[[package]] +name = "unic-char-property" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8c57a407d9b6fa02b4795eb81c5b6652060a15a7903ea981f3d723e6c0be221" +dependencies = [ + "unic-char-range", +] + +[[package]] +name = "unic-char-range" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0398022d5f700414f6b899e10b8348231abf9173fa93144cbc1a43b9793c1fbc" + +[[package]] +name = "unic-common" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "80d7ff825a6a654ee85a63e80f92f054f904f21e7d12da4e22f9834a4aaa35bc" + +[[package]] +name = "unic-ucd-ident" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e230a37c0381caa9219d67cf063aa3a375ffed5bf541a452db16e744bdab6987" +dependencies = [ + "unic-char-property", + "unic-char-range", + "unic-ucd-version", +] + +[[package]] +name = "unic-ucd-version" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96bd2f2237fe450fcd0a1d2f5f4e91711124f7857ba2e964247776ebeeb7b0c4" +dependencies = [ + "unic-common", +] + +[[package]] +name = "unicode-ident" +version = "1.0.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75" + +[[package]] +name = "unicode-segmentation" +version = "1.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9629274872b2bfaf8d66f5f15725007f635594914870f65218920345aa11aa8c" + +[[package]] +name = "unicode-xid" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebc1c04c71510c7f702b52b7c350734c9ff1295c464a03335b00bb84fc54f853" + +[[package]] +name = "universal-hash" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fc1de2c688dc15305988b563c3854064043356019f97a4b46276fe734c4f07ea" +dependencies = [ + "crypto-common", + "subtle", +] + +[[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_iter" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be" + +[[package]] +name = "uuid" +version = "1.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ac8b6f42ead25368cf5b098aeb3dc8a1a2c05a3eee8a9a1a68c640edbfc79d9" +dependencies = [ + "getrandom 0.4.2", + "js-sys", + "serde_core", + "wasm-bindgen", +] + +[[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 = "wasip3" +version = "0.4.0+wasi-0.3.0-rc-2026-01-06" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5428f8bf88ea5ddc08faddef2ac4a67e390b88186c703ce6dbd955e1c145aca5" +dependencies = [ + "wit-bindgen", +] + +[[package]] +name = "wasm-bindgen" +version = "0.2.118" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bf938a0bacb0469e83c1e148908bd7d5a6010354cf4fb73279b7447422e3a89" +dependencies = [ + "cfg-if", + "once_cell", + "rustversion", + "wasm-bindgen-macro", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.68" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f371d383f2fb139252e0bfac3b81b265689bf45b6874af544ffa4c975ac1ebf8" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.118" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eeff24f84126c0ec2db7a449f0c2ec963c6a49efe0698c4242929da037ca28ed" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.118" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d08065faf983b2b80a79fd87d8254c409281cf7de75fc4b773019824196c904" +dependencies = [ + "bumpalo", + "proc-macro2", + "quote", + "syn 2.0.117", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.118" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5fd04d9e306f1907bd13c6361b5c6bfc7b3b3c095ed3f8a9246390f8dbdee129" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "wasm-encoder" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "990065f2fe63003fe337b932cfb5e3b80e0b4d0f5ff650e6985b1048f62c8319" +dependencies = [ + "leb128fmt", + "wasmparser", +] + +[[package]] +name = "wasm-metadata" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bb0e353e6a2fbdc176932bbaab493762eb1255a7900fe0fea1a2f96c296cc909" +dependencies = [ + "anyhow", + "indexmap 2.14.0", + "wasm-encoder", + "wasmparser", +] + +[[package]] +name = "wasm-streams" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d1ec4f6517c9e11ae630e200b2b65d193279042e28edd4a2cda233e46670bbb" +dependencies = [ + "futures-util", + "js-sys", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "wasmparser" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47b807c72e1bac69382b3a6fb3dbe8ea4c0ed87ff5629b8685ae6b9a611028fe" +dependencies = [ + "bitflags 2.11.0", + "hashbrown 0.15.5", + "indexmap 2.14.0", + "semver", +] + +[[package]] +name = "web-sys" +version = "0.3.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f2dfbb17949fa2088e5d39408c48368947b86f7834484e87b73de55bc14d97d" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "web_atoms" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "57a9779e9f04d2ac1ce317aee707aa2f6b773afba7b931222bff6983843b1576" +dependencies = [ + "phf 0.13.1", + "phf_codegen 0.13.1", + "string_cache 0.9.0", + "string_cache_codegen 0.6.1", +] + +[[package]] +name = "webkit2gtk" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1027150013530fb2eaf806408df88461ae4815a45c541c8975e61d6f2fc4793" +dependencies = [ + "bitflags 1.3.2", + "cairo-rs", + "gdk", + "gdk-sys", + "gio", + "gio-sys", + "glib", + "glib-sys", + "gobject-sys", + "gtk", + "gtk-sys", + "javascriptcore-rs", + "libc", + "once_cell", + "soup3", + "webkit2gtk-sys", +] + +[[package]] +name = "webkit2gtk-sys" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "916a5f65c2ef0dfe12fff695960a2ec3d4565359fdbb2e9943c974e06c734ea5" +dependencies = [ + "bitflags 1.3.2", + "cairo-sys-rs", + "gdk-sys", + "gio-sys", + "glib-sys", + "gobject-sys", + "gtk-sys", + "javascriptcore-rs-sys", + "libc", + "pkg-config", + "soup3-sys", + "system-deps", +] + +[[package]] +name = "webview2-com" +version = "0.38.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7130243a7a5b33c54a444e54842e6a9e133de08b5ad7b5861cd8ed9a6a5bc96a" +dependencies = [ + "webview2-com-macros", + "webview2-com-sys", + "windows", + "windows-core 0.61.2", + "windows-implement", + "windows-interface", +] + +[[package]] +name = "webview2-com-macros" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67a921c1b6914c367b2b823cd4cde6f96beec77d30a939c8199bb377cf9b9b54" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "webview2-com-sys" +version = "0.38.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "381336cfffd772377d291702245447a5251a2ffa5bad679c99e61bc48bacbf9c" +dependencies = [ + "thiserror 2.0.18", + "windows", + "windows-core 0.61.2", +] + +[[package]] +name = "winapi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" +dependencies = [ + "winapi-i686-pc-windows-gnu", + "winapi-x86_64-pc-windows-gnu", +] + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" + +[[package]] +name = "winapi-util" +version = "0.1.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2a7b1c03c876122aa43f3020e6c3c3ee5c05081c9a00739faf7503aeba10d22" +dependencies = [ + "windows-sys 0.61.2", +] + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" + +[[package]] +name = "window-vibrancy" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9bec5a31f3f9362f2258fd0e9c9dd61a9ca432e7306cc78c444258f0dce9a9c" +dependencies = [ + "objc2", + "objc2-app-kit", + "objc2-core-foundation", + "objc2-foundation", + "raw-window-handle", + "windows-sys 0.59.0", + "windows-version", +] + +[[package]] +name = "windows" +version = "0.61.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9babd3a767a4c1aef6900409f85f5d53ce2544ccdfaa86dad48c91782c6d6893" +dependencies = [ + "windows-collections", + "windows-core 0.61.2", + "windows-future", + "windows-link 0.1.3", + "windows-numerics", +] + +[[package]] +name = "windows-collections" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3beeceb5e5cfd9eb1d76b381630e82c4241ccd0d27f1a39ed41b2760b255c5e8" +dependencies = [ + "windows-core 0.61.2", +] + +[[package]] +name = "windows-core" +version = "0.61.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c0fdd3ddb90610c7638aa2b3a3ab2904fb9e5cdbecc643ddb3647212781c4ae3" +dependencies = [ + "windows-implement", + "windows-interface", + "windows-link 0.1.3", + "windows-result 0.3.4", + "windows-strings 0.4.2", +] + +[[package]] +name = "windows-core" +version = "0.62.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8e83a14d34d0623b51dce9581199302a221863196a1dde71a7663a4c2be9deb" +dependencies = [ + "windows-implement", + "windows-interface", + "windows-link 0.2.1", + "windows-result 0.4.1", + "windows-strings 0.5.1", +] + +[[package]] +name = "windows-future" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fc6a41e98427b19fe4b73c550f060b59fa592d7d686537eebf9385621bfbad8e" +dependencies = [ + "windows-core 0.61.2", + "windows-link 0.1.3", + "windows-threading", +] + +[[package]] +name = "windows-implement" +version = "0.60.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "053e2e040ab57b9dc951b72c264860db7eb3b0200ba345b4e4c3b14f67855ddf" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "windows-interface" +version = "0.59.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f316c4a2570ba26bbec722032c4099d8c8bc095efccdc15688708623367e358" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "windows-link" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e6ad25900d524eaabdbbb96d20b4311e1e7ae1699af4fb28c17ae66c80d798a" + +[[package]] +name = "windows-link" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0805222e57f7521d6a62e36fa9163bc891acd422f971defe97d64e70d0a4fe5" + +[[package]] +name = "windows-numerics" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9150af68066c4c5c07ddc0ce30421554771e528bde427614c61038bc2c92c2b1" +dependencies = [ + "windows-core 0.61.2", + "windows-link 0.1.3", +] + +[[package]] +name = "windows-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.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df79d97927682d2fd8adb29682d1140b343be4ac0f08fd68b7765d9c059d3945" + +[[package]] +name = "winnow" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09dac053f1cd375980747450bfc7250c264eaae0583872e845c0c7cd578872b5" +dependencies = [ + "memchr", +] + +[[package]] +name = "winreg" +version = "0.55.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb5a765337c50e9ec252c2069be9bf91c7df47afb103b642ba3a53bf8101be97" +dependencies = [ + "cfg-if", + "windows-sys 0.59.0", +] + +[[package]] +name = "wit-bindgen" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7249219f66ced02969388cf2bb044a09756a083d0fab1e566056b04d9fbcaa5" +dependencies = [ + "wit-bindgen-rust-macro", +] + +[[package]] +name = "wit-bindgen-core" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ea61de684c3ea68cb082b7a88508a8b27fcc8b797d738bfc99a82facf1d752dc" +dependencies = [ + "anyhow", + "heck 0.5.0", + "wit-parser", +] + +[[package]] +name = "wit-bindgen-rust" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7c566e0f4b284dd6561c786d9cb0142da491f46a9fbed79ea69cdad5db17f21" +dependencies = [ + "anyhow", + "heck 0.5.0", + "indexmap 2.14.0", + "prettyplease", + "syn 2.0.117", + "wasm-metadata", + "wit-bindgen-core", + "wit-component", +] + +[[package]] +name = "wit-bindgen-rust-macro" +version = "0.51.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c0f9bfd77e6a48eccf51359e3ae77140a7f50b1e2ebfe62422d8afdaffab17a" +dependencies = [ + "anyhow", + "prettyplease", + "proc-macro2", + "quote", + "syn 2.0.117", + "wit-bindgen-core", + "wit-bindgen-rust", +] + +[[package]] +name = "wit-component" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d66ea20e9553b30172b5e831994e35fbde2d165325bec84fc43dbf6f4eb9cb2" +dependencies = [ + "anyhow", + "bitflags 2.11.0", + "indexmap 2.14.0", + "log", + "serde", + "serde_derive", + "serde_json", + "wasm-encoder", + "wasm-metadata", + "wasmparser", + "wit-parser", +] + +[[package]] +name = "wit-parser" +version = "0.244.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ecc8ac4bc1dc3381b7f59c34f00b67e18f910c2c0f50015669dde7def656a736" +dependencies = [ + "anyhow", + "id-arena", + "indexmap 2.14.0", + "log", + "semver", + "serde", + "serde_derive", + "serde_json", + "unicode-xid", + "wasmparser", +] + +[[package]] +name = "writeable" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1ffae5123b2d3fc086436f8834ae3ab053a283cfac8fe0a0b8eaae044768a4c4" + +[[package]] +name = "wry" +version = "0.54.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5a8135d8676225e5744de000d4dff5a082501bf7db6a1c1495034f8c314edbc" +dependencies = [ + "base64 0.22.1", + "block2", + "cookie", + "crossbeam-channel", + "dirs 6.0.0", + "dom_query", + "dpi", + "dunce", + "gdkx11", + "gtk", + "http", + "javascriptcore-rs", + "jni", + "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 = "x11" +version = "2.21.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "502da5464ccd04011667b11c435cb992822c2c0dbde1770c988480d312a0db2e" +dependencies = [ + "libc", + "pkg-config", +] + +[[package]] +name = "x11-dl" +version = "2.21.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38735924fedd5314a6e548792904ed8c6de6636285cb9fec04d5b1db85c1516f" +dependencies = [ + "libc", + "once_cell", + "pkg-config", +] + +[[package]] +name = "yoke" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "abe8c5fda708d9ca3df187cae8bfb9ceda00dd96231bed36e445a1a48e66f9ca" +dependencies = [ + "stable_deref_trait", + "yoke-derive", + "zerofrom", +] + +[[package]] +name = "yoke-derive" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "de844c262c8848816172cef550288e7dc6c7b7814b4ee56b3e1553f275f1858e" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", + "synstructure", +] + +[[package]] +name = "zerocopy" +version = "0.8.48" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eed437bf9d6692032087e337407a86f04cd8d6a16a37199ed57949d415bd68e9" +dependencies = [ + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.8.48" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70e3cd084b1788766f53af483dd21f93881ff30d7320490ec3ef7526d203bad4" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "zerofrom" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69faa1f2a1ea75661980b013019ed6687ed0e83d069bc1114e2cc74c6c04c4df" +dependencies = [ + "zerofrom-derive", +] + +[[package]] +name = "zerofrom-derive" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11532158c46691caf0f2593ea8358fed6bbf68a0315e80aae9bd41fbade684a1" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", + "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.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f9152d31db0792fa83f70fb2f83148effb5c1f5b8c7686c3459e361d9bc20bf" +dependencies = [ + "displaydoc", + "yoke", + "zerofrom", +] + +[[package]] +name = "zerovec" +version = "0.11.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90f911cbc359ab6af17377d242225f4d75119aec87ea711a880987b18cd7b239" +dependencies = [ + "yoke", + "zerofrom", + "zerovec-derive", +] + +[[package]] +name = "zerovec-derive" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "625dc425cab0dca6dc3c3319506e6593dcb08a9f387ea3b284dbd52a92c40555" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "zmij" +version = "1.0.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8848ee67ecc8aedbaf3e4122217aff892639231befc6a1b58d29fff4c2cabaa" diff --git a/src-tauri/Cargo.toml b/src-tauri/Cargo.toml new file mode 100644 index 0000000..9466249 --- /dev/null +++ b/src-tauri/Cargo.toml @@ -0,0 +1,43 @@ +[package] +name = "orchai" +version = "0.1.0" +description = "Orchai - Tuleap tracker monitor & AI agent dispatcher" +authors = [] +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[lib] +# The `_lib` suffix may seem redundant but it is necessary +# to make the lib name unique and wouldn't conflict with the bin name. +# This seems to be only an issue on Windows, see https://github.com/rust-lang/cargo/issues/8519 +name = "orchai_lib" +crate-type = ["staticlib", "cdylib", "rlib"] + +[build-dependencies] +tauri-build = { version = "2", features = [] } + +[dependencies] +tauri = { version = "2", features = [] } +tauri-plugin-dialog = "2" +serde = { version = "1", features = ["derive"] } +serde_json = "1" +rusqlite = { version = "0.31", features = ["bundled"] } +uuid = { version = "1", features = ["v4", "serde"] } +chrono = { version = "0.4", features = ["serde"] } +dirs = "5" +reqwest = { version = "0.12", features = ["json"] } +tokio = { version = "1", features = ["time", "sync", "macros"] } +aes-gcm = "0.10" +rand = "0.8" +base64 = "0.22" + +[profile.dev] +incremental = true # Compiles your binary in smaller steps. + +[profile.release] +codegen-units = 1 # Allows LLVM to perform better optimization. +lto = true # Enables link-time-optimizations. +opt-level = "s" # Prioritizes small binary size. Use `3` if you prefer speed. +panic = "abort" # Higher performance by disabling panic handlers. +strip = true # Ensures debug symbols are removed. 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/capabilities/default.json b/src-tauri/capabilities/default.json new file mode 100644 index 0000000..e895c6b --- /dev/null +++ b/src-tauri/capabilities/default.json @@ -0,0 +1,10 @@ +{ + "$schema": "../gen/schemas/desktop-schema.json", + "identifier": "default", + "description": "Capability for the main window", + "windows": ["main"], + "permissions": [ + "core:default", + "dialog:default" + ] +} diff --git a/src-tauri/icons/128x128.png b/src-tauri/icons/128x128.png new file mode 100644 index 0000000..4e3b849 Binary files /dev/null and b/src-tauri/icons/128x128.png differ diff --git a/src-tauri/icons/128x128@2x.png b/src-tauri/icons/128x128@2x.png new file mode 100644 index 0000000..ca4458a Binary files /dev/null and b/src-tauri/icons/128x128@2x.png differ diff --git a/src-tauri/icons/32x32.png b/src-tauri/icons/32x32.png new file mode 100644 index 0000000..7c5c6e3 Binary files /dev/null and b/src-tauri/icons/32x32.png differ diff --git a/src-tauri/icons/icon.icns b/src-tauri/icons/icon.icns new file mode 100644 index 0000000..6d78c7b Binary files /dev/null and b/src-tauri/icons/icon.icns differ diff --git a/src-tauri/icons/icon.ico b/src-tauri/icons/icon.ico new file mode 100644 index 0000000..8d90e39 Binary files /dev/null and b/src-tauri/icons/icon.ico differ diff --git a/src-tauri/migrations/001_init.sql b/src-tauri/migrations/001_init.sql new file mode 100644 index 0000000..ea1a332 --- /dev/null +++ b/src-tauri/migrations/001_init.sql @@ -0,0 +1,63 @@ +CREATE TABLE IF NOT EXISTS projects ( + id TEXT PRIMARY KEY, + name TEXT NOT NULL, + path TEXT NOT NULL, + cloned_from TEXT, + base_branch TEXT NOT NULL DEFAULT 'main', + created_at TEXT NOT NULL DEFAULT (datetime('now')) +); + +CREATE TABLE IF NOT EXISTS tuleap_credentials ( + id TEXT PRIMARY KEY, + tuleap_url TEXT NOT NULL, + username TEXT NOT NULL, + password_encrypted TEXT NOT NULL +); + +CREATE TABLE IF NOT EXISTS watched_trackers ( + id TEXT PRIMARY KEY, + project_id TEXT NOT NULL REFERENCES projects(id) ON DELETE CASCADE, + tracker_id INTEGER NOT NULL, + tracker_label TEXT NOT NULL, + polling_interval INTEGER NOT NULL DEFAULT 10, + agent_config_json TEXT NOT NULL, + filters_json TEXT NOT NULL, + created_at TEXT NOT NULL DEFAULT (datetime('now')) +); + +CREATE TABLE IF NOT EXISTS processed_tickets ( + id TEXT PRIMARY KEY, + tracker_id TEXT NOT NULL REFERENCES watched_trackers(id) ON DELETE CASCADE, + artifact_id INTEGER NOT NULL, + artifact_title TEXT NOT NULL, + artifact_data TEXT NOT NULL, + status TEXT NOT NULL DEFAULT 'Pending', + analyst_report TEXT, + developer_report TEXT, + worktree_path TEXT, + branch_name TEXT, + detected_at TEXT NOT NULL DEFAULT (datetime('now')), + processed_at TEXT +); + +CREATE TABLE IF NOT EXISTS worktrees ( + id TEXT PRIMARY KEY, + ticket_id TEXT NOT NULL REFERENCES processed_tickets(id), + path TEXT NOT NULL, + branch_name TEXT NOT NULL, + status TEXT NOT NULL DEFAULT 'Active', + created_at TEXT NOT NULL DEFAULT (datetime('now')), + merged_at TEXT, + merged_into TEXT +); + +CREATE TABLE IF NOT EXISTS notifications ( + id TEXT PRIMARY KEY, + project_id TEXT NOT NULL REFERENCES projects(id) ON DELETE CASCADE, + ticket_id TEXT REFERENCES processed_tickets(id), + type TEXT NOT NULL, + title TEXT NOT NULL, + message TEXT NOT NULL, + read INTEGER NOT NULL DEFAULT 0, + created_at TEXT NOT NULL DEFAULT (datetime('now')) +); diff --git a/src-tauri/migrations/002_add_last_polled.sql b/src-tauri/migrations/002_add_last_polled.sql new file mode 100644 index 0000000..da2cc58 --- /dev/null +++ b/src-tauri/migrations/002_add_last_polled.sql @@ -0,0 +1,2 @@ +ALTER TABLE watched_trackers ADD COLUMN last_polled_at TEXT; +ALTER TABLE watched_trackers ADD COLUMN enabled INTEGER NOT NULL DEFAULT 1; diff --git a/src-tauri/src/commands/credential.rs b/src-tauri/src/commands/credential.rs new file mode 100644 index 0000000..954c9f9 --- /dev/null +++ b/src-tauri/src/commands/credential.rs @@ -0,0 +1,80 @@ +use crate::error::AppError; +use crate::models::credential::{TuleapCredentials, TuleapCredentialsSafe}; +use crate::services::crypto; +use crate::AppState; +use tauri::State; + +#[tauri::command] +pub fn set_tuleap_credentials( + state: State<'_, AppState>, + tuleap_url: String, + username: String, + password: String, +) -> Result { + let password_encrypted = crypto::encrypt(&state.encryption_key, &password) + .map_err(AppError::from)?; + + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let creds = TuleapCredentials::upsert(&db, &tuleap_url, &username, &password_encrypted)?; + Ok(creds.to_safe()) +} + +#[tauri::command] +pub fn get_tuleap_credentials( + state: State<'_, AppState>, +) -> Result, AppError> { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let result = TuleapCredentials::get(&db)?.map(|c| c.to_safe()); + Ok(result) +} + +#[tauri::command] +pub fn delete_tuleap_credentials(state: State<'_, AppState>) -> Result<(), AppError> { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + TuleapCredentials::delete(&db)?; + Ok(()) +} + +#[tauri::command] +pub async fn test_tuleap_connection(state: State<'_, AppState>) -> Result { + let (tuleap_url, username, password) = { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let creds = TuleapCredentials::get(&db)? + .ok_or_else(|| AppError::from("No credentials configured".to_string()))?; + + let password = crypto::decrypt(&state.encryption_key, &creds.password_encrypted) + .map_err(AppError::from)?; + + (creds.tuleap_url, creds.username, password) + }; + + let url = format!("{}/api/projects?limit=1", tuleap_url.trim_end_matches('/')); + + state + .http_client + .get(&url) + .basic_auth(&username, Some(&password)) + .send() + .await + .map_err(AppError::from)? + .error_for_status() + .map_err(AppError::from)?; + + Ok("Connection successful".to_string()) +} diff --git a/src-tauri/src/commands/mod.rs b/src-tauri/src/commands/mod.rs new file mode 100644 index 0000000..e404626 --- /dev/null +++ b/src-tauri/src/commands/mod.rs @@ -0,0 +1,4 @@ +pub mod credential; +pub mod poller; +pub mod project; +pub mod tracker; diff --git a/src-tauri/src/commands/poller.rs b/src-tauri/src/commands/poller.rs new file mode 100644 index 0000000..6290e3b --- /dev/null +++ b/src-tauri/src/commands/poller.rs @@ -0,0 +1,98 @@ +use crate::error::AppError; +use crate::models::credential::TuleapCredentials; +use crate::models::ticket::ProcessedTicket; +use crate::models::tracker::WatchedTracker; +use crate::services::{crypto, filter_engine}; +use crate::services::tuleap_client::TuleapClient; +use crate::AppState; +use tauri::State; + +#[tauri::command] +pub async fn manual_poll( + state: State<'_, AppState>, + tracker_id: String, +) -> Result, AppError> { + let (tracker, client) = { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let tracker = WatchedTracker::get_by_id(&db, &tracker_id)?; + + let cred = TuleapCredentials::get(&db)? + .ok_or_else(|| AppError::from("No Tuleap credentials configured".to_string()))?; + + let password = crypto::decrypt(&state.encryption_key, &cred.password_encrypted) + .map_err(AppError::from)?; + + let client = TuleapClient::new( + &state.http_client, + &cred.tuleap_url, + &cred.username, + &password, + ); + + (tracker, client) + }; // lock dropped here + + let artifacts = client + .get_artifacts(tracker.tracker_id) + .await + .map_err(AppError::from)?; + + let filtered = filter_engine::apply_filters(&artifacts, &tracker.filters); + + let mut newly_inserted: Vec = Vec::new(); + + { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + for artifact in &filtered { + let artifact_id = artifact + .get("id") + .and_then(|v| v.as_i64()) + .unwrap_or(0) as i32; + + let artifact_title = artifact + .get("title") + .and_then(|v| v.as_str()) + .unwrap_or("") + .to_string(); + + let artifact_data = serde_json::to_string(artifact) + .unwrap_or_else(|_| "{}".to_string()); + + if let Some(ticket) = ProcessedTicket::insert_if_new( + &db, + &tracker.id, + artifact_id, + &artifact_title, + &artifact_data, + )? { + newly_inserted.push(ticket); + } + } + + WatchedTracker::update_last_polled(&db, &tracker.id)?; + } + + Ok(newly_inserted) +} + +#[tauri::command] +pub fn get_queue_status( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let tickets = ProcessedTicket::list_by_project(&db, &project_id)?; + Ok(tickets) +} diff --git a/src-tauri/src/commands/project.rs b/src-tauri/src/commands/project.rs new file mode 100644 index 0000000..8e744ac --- /dev/null +++ b/src-tauri/src/commands/project.rs @@ -0,0 +1,83 @@ +use crate::error::AppError; +use crate::models::project::Project; +use crate::AppState; +use std::process::Command; +use tauri::State; + +#[tauri::command] +pub fn create_project( + state: State<'_, AppState>, + name: String, + path_or_url: String, + base_branch: String, +) -> Result { + let is_url = path_or_url.starts_with("http://") + || path_or_url.starts_with("https://") + || path_or_url.starts_with("git@"); + + let (local_path, cloned_from) = if is_url { + let home = dirs::home_dir().ok_or_else(|| AppError::from("Cannot determine home directory".to_string()))?; + let clone_dir = home.join("orchai-repos").join(&name); + std::fs::create_dir_all(&clone_dir)?; + + let clone_dir_str = clone_dir.to_str() + .ok_or_else(|| AppError::from("Clone path contains invalid characters".to_string()))?; + + let output = Command::new("git") + .args(["clone", &path_or_url, clone_dir_str]) + .output()?; + + if !output.status.success() { + let stderr = String::from_utf8_lossy(&output.stderr); + return Err(AppError::from(format!("git clone failed: {}", stderr))); + } + + (clone_dir.to_string_lossy().to_string(), Some(path_or_url)) + } else { + let path = std::path::Path::new(&path_or_url); + if !path.exists() { + return Err(AppError::from(format!("Path does not exist: {}", path_or_url))); + } + if !path.join(".git").exists() { + return Err(AppError::from(format!("Not a git repository: {}", path_or_url))); + } + (path_or_url, None) + }; + + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let project = Project::insert(&db, &name, &local_path, cloned_from.as_deref(), &base_branch)?; + Ok(project) +} + +#[tauri::command] +pub fn list_projects(state: State<'_, AppState>) -> Result, AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let projects = Project::list(&db)?; + Ok(projects) +} + +#[tauri::command] +pub fn get_project(state: State<'_, AppState>, id: String) -> Result { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + let project = Project::get_by_id(&db, &id)?; + Ok(project) +} + +#[tauri::command] +pub fn update_project( + state: State<'_, AppState>, + id: String, + name: String, + base_branch: String, +) -> Result<(), AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + Project::update(&db, &id, &name, &base_branch)?; + Ok(()) +} + +#[tauri::command] +pub fn delete_project(state: State<'_, AppState>, id: String) -> Result<(), AppError> { + let db = state.db.lock().map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + Project::delete(&db, &id)?; + Ok(()) +} diff --git a/src-tauri/src/commands/tracker.rs b/src-tauri/src/commands/tracker.rs new file mode 100644 index 0000000..3784062 --- /dev/null +++ b/src-tauri/src/commands/tracker.rs @@ -0,0 +1,126 @@ +use crate::error::AppError; +use crate::models::credential::TuleapCredentials; +use crate::models::ticket::ProcessedTicket; +use crate::models::tracker::{AgentConfig, FilterGroup, WatchedTracker}; +use crate::services::crypto; +use crate::services::tuleap_client::TuleapClient; +use crate::AppState; +use tauri::State; + +fn build_tuleap_client(state: &State) -> Result { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let cred = TuleapCredentials::get(&db)? + .ok_or_else(|| AppError::from("No Tuleap credentials configured".to_string()))?; + + let password = crypto::decrypt(&state.encryption_key, &cred.password_encrypted) + .map_err(AppError::from)?; + + Ok(TuleapClient::new( + &state.http_client, + &cred.tuleap_url, + &cred.username, + &password, + )) +} + +#[tauri::command] +pub fn add_tracker( + state: State<'_, AppState>, + project_id: String, + tracker_id: i32, + tracker_label: String, + polling_interval: i32, + agent_config: AgentConfig, + filters: Vec, +) -> Result { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let tracker = WatchedTracker::insert( + &db, + &project_id, + tracker_id, + &tracker_label, + polling_interval, + agent_config, + filters, + )?; + + Ok(tracker) +} + +#[tauri::command] +pub fn list_trackers( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let trackers = WatchedTracker::list_by_project(&db, &project_id)?; + Ok(trackers) +} + +#[tauri::command] +pub fn update_tracker( + state: State<'_, AppState>, + id: String, + polling_interval: i32, + agent_config: AgentConfig, + filters: Vec, + enabled: bool, +) -> Result<(), AppError> { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + WatchedTracker::update(&db, &id, polling_interval, agent_config, filters, enabled)?; + Ok(()) +} + +#[tauri::command] +pub fn remove_tracker(state: State<'_, AppState>, id: String) -> Result<(), AppError> { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + WatchedTracker::delete(&db, &id)?; + Ok(()) +} + +#[tauri::command] +pub async fn get_tracker_fields( + state: State<'_, AppState>, + tracker_id: i32, +) -> Result, AppError> { + let client = build_tuleap_client(&state)?; + let fields = client + .get_tracker_fields(tracker_id) + .await + .map_err(AppError::from)?; + Ok(fields) +} + +#[tauri::command] +pub fn list_processed_tickets( + state: State<'_, AppState>, + project_id: String, +) -> Result, AppError> { + let db = state + .db + .lock() + .map_err(|e| AppError::from(format!("Database lock failed: {}", e)))?; + + let tickets = ProcessedTicket::list_by_project(&db, &project_id)?; + Ok(tickets) +} diff --git a/src-tauri/src/db.rs b/src-tauri/src/db.rs new file mode 100644 index 0000000..8ceb7f7 --- /dev/null +++ b/src-tauri/src/db.rs @@ -0,0 +1,91 @@ +use rusqlite::{Connection, Result}; +use std::path::Path; + +const MIGRATION_001: &str = include_str!("../migrations/001_init.sql"); +const MIGRATION_002: &str = include_str!("../migrations/002_add_last_polled.sql"); + +pub fn init(db_path: &Path) -> Result { + let conn = Connection::open(db_path)?; + configure(&conn)?; + migrate(&conn)?; + Ok(conn) +} + +#[cfg(test)] +pub fn init_in_memory() -> Result { + let conn = Connection::open_in_memory()?; + configure(&conn)?; + migrate(&conn)?; + Ok(conn) +} + +fn configure(conn: &Connection) -> Result<()> { + conn.pragma_update(None, "journal_mode", "wal")?; + conn.pragma_update(None, "foreign_keys", "ON")?; + Ok(()) +} + +fn migrate(conn: &Connection) -> Result<()> { + let version: i32 = conn.pragma_query_value(None, "user_version", |row| row.get(0))?; + + if version < 1 { + conn.execute_batch(MIGRATION_001)?; + conn.pragma_update(None, "user_version", 1)?; + } + if version < 2 { + conn.execute_batch(MIGRATION_002)?; + conn.pragma_update(None, "user_version", 2)?; + } + + Ok(()) +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_init_in_memory_creates_tables() { + let conn = init_in_memory().expect("should initialize"); + + // Verify all 6 tables exist + let tables: Vec = conn + .prepare("SELECT name FROM sqlite_master WHERE type='table' AND name NOT LIKE 'sqlite_%' ORDER BY name") + .unwrap() + .query_map([], |row| row.get(0)) + .unwrap() + .collect::, _>>() + .unwrap(); + + assert_eq!( + tables, + vec![ + "notifications", + "processed_tickets", + "projects", + "tuleap_credentials", + "watched_trackers", + "worktrees", + ] + ); + } + + #[test] + fn test_init_in_memory_enables_foreign_keys() { + let conn = init_in_memory().expect("should initialize"); + let fk_enabled: i32 = conn + .query_row("PRAGMA foreign_keys", [], |row| row.get(0)) + .unwrap(); + assert_eq!(fk_enabled, 1); + } + + #[test] + fn test_migration_is_idempotent() { + let conn = init_in_memory().expect("should initialize"); + // Running init again on same connection should not fail + let version: i32 = conn + .pragma_query_value(None, "user_version", |row| row.get(0)) + .unwrap(); + assert_eq!(version, 2); + } +} diff --git a/src-tauri/src/error.rs b/src-tauri/src/error.rs new file mode 100644 index 0000000..914ed59 --- /dev/null +++ b/src-tauri/src/error.rs @@ -0,0 +1,42 @@ +use serde::Serialize; + +#[derive(Debug, Serialize)] +pub struct AppError { + pub message: String, +} + +impl From for AppError { + fn from(e: rusqlite::Error) -> Self { + AppError { + message: e.to_string(), + } + } +} + +impl From for AppError { + fn from(e: std::io::Error) -> Self { + AppError { + message: e.to_string(), + } + } +} + +impl From for AppError { + fn from(s: String) -> Self { + AppError { message: s } + } +} + +impl From for AppError { + fn from(e: reqwest::Error) -> Self { + AppError { message: e.to_string() } + } +} + +impl std::fmt::Display for AppError { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + write!(f, "{}", self.message) + } +} + +impl std::error::Error for AppError {} diff --git a/src-tauri/src/lib.rs b/src-tauri/src/lib.rs new file mode 100644 index 0000000..c735851 --- /dev/null +++ b/src-tauri/src/lib.rs @@ -0,0 +1,88 @@ +mod commands; +mod db; +mod error; +mod models; +mod services; + +use std::sync::{Arc, Mutex}; +use tauri::Manager; + +pub struct AppState { + pub db: Arc>, + pub encryption_key: [u8; 32], + pub http_client: reqwest::Client, +} + +#[cfg_attr(mobile, tauri::mobile_entry_point)] +pub fn run() { + tauri::Builder::default() + .plugin(tauri_plugin_dialog::init()) + .setup(|app| { + let db_dir = app.path().app_data_dir()?; + std::fs::create_dir_all(&db_dir)?; + + let db_path = db_dir.join("orchai.db"); + let conn = db::init(&db_path).expect("Failed to initialize database"); + + let key_path = db_dir.join("orchai.key"); + let encryption_key = load_or_generate_key(&key_path)?; + + let http_client = reqwest::Client::new(); + + let db_arc = Arc::new(Mutex::new(conn)); + app.manage(AppState { + db: db_arc.clone(), + encryption_key, + http_client: http_client.clone(), + }); + + // Start background poller + services::poller::start( + db_arc, + encryption_key, + http_client, + app.handle().clone(), + ); + + Ok(()) + }) + .invoke_handler(tauri::generate_handler![ + commands::project::create_project, + commands::project::list_projects, + commands::project::get_project, + commands::project::update_project, + commands::project::delete_project, + commands::credential::set_tuleap_credentials, + commands::credential::get_tuleap_credentials, + commands::credential::delete_tuleap_credentials, + commands::credential::test_tuleap_connection, + commands::tracker::add_tracker, + commands::tracker::list_trackers, + commands::tracker::update_tracker, + commands::tracker::remove_tracker, + commands::tracker::get_tracker_fields, + commands::tracker::list_processed_tickets, + commands::poller::manual_poll, + commands::poller::get_queue_status, + ]) + .run(tauri::generate_context!()) + .expect("error while running tauri application"); +} + +fn load_or_generate_key(path: &std::path::Path) -> Result<[u8; 32], Box> { + use rand::RngCore; + if path.exists() { + let bytes = std::fs::read(path)?; + if bytes.len() != 32 { + return Err("Invalid key file size".into()); + } + let mut key = [0u8; 32]; + key.copy_from_slice(&bytes); + Ok(key) + } else { + let mut key = [0u8; 32]; + rand::rngs::OsRng.fill_bytes(&mut key); + std::fs::write(path, key)?; + Ok(key) + } +} diff --git a/src-tauri/src/main.rs b/src-tauri/src/main.rs new file mode 100644 index 0000000..93b3227 --- /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() { + orchai_lib::run() +} diff --git a/src-tauri/src/models/credential.rs b/src-tauri/src/models/credential.rs new file mode 100644 index 0000000..0849947 --- /dev/null +++ b/src-tauri/src/models/credential.rs @@ -0,0 +1,165 @@ +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct TuleapCredentials { + pub id: String, + pub tuleap_url: String, + pub username: String, + pub password_encrypted: String, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct TuleapCredentialsSafe { + pub id: String, + pub tuleap_url: String, + pub username: String, +} + +impl TuleapCredentials { + pub fn upsert( + conn: &Connection, + tuleap_url: &str, + username: &str, + password_encrypted: &str, + ) -> Result { + conn.execute("DELETE FROM tuleap_credentials", [])?; + + let id = Uuid::new_v4().to_string(); + conn.execute( + "INSERT INTO tuleap_credentials (id, tuleap_url, username, password_encrypted) VALUES (?1, ?2, ?3, ?4)", + params![id, tuleap_url, username, password_encrypted], + )?; + + Ok(TuleapCredentials { + id, + tuleap_url: tuleap_url.to_string(), + username: username.to_string(), + password_encrypted: password_encrypted.to_string(), + }) + } + + pub fn get(conn: &Connection) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, tuleap_url, username, password_encrypted FROM tuleap_credentials LIMIT 1", + )?; + let mut rows = stmt.query_map([], |row| { + Ok(TuleapCredentials { + id: row.get(0)?, + tuleap_url: row.get(1)?, + username: row.get(2)?, + password_encrypted: row.get(3)?, + }) + })?; + + match rows.next() { + Some(row) => Ok(Some(row?)), + None => Ok(None), + } + } + + pub fn delete(conn: &Connection) -> Result<()> { + conn.execute("DELETE FROM tuleap_credentials", [])?; + Ok(()) + } + + pub fn to_safe(&self) -> TuleapCredentialsSafe { + TuleapCredentialsSafe { + id: self.id.clone(), + tuleap_url: self.tuleap_url.clone(), + username: self.username.clone(), + } + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + + fn setup() -> Connection { + db::init_in_memory().expect("db init should succeed") + } + + #[test] + fn test_upsert_creates_credentials() { + let conn = setup(); + let creds = TuleapCredentials::upsert( + &conn, + "https://tuleap.example.com", + "alice", + "encrypted_password", + ) + .expect("upsert should succeed"); + + assert_eq!(creds.tuleap_url, "https://tuleap.example.com"); + assert_eq!(creds.username, "alice"); + assert_eq!(creds.password_encrypted, "encrypted_password"); + assert!(!creds.id.is_empty()); + } + + #[test] + fn test_upsert_replaces_existing() { + let conn = setup(); + TuleapCredentials::upsert(&conn, "https://old.example.com", "old_user", "old_enc") + .expect("first upsert should succeed"); + + let second = TuleapCredentials::upsert( + &conn, + "https://new.example.com", + "new_user", + "new_enc", + ) + .expect("second upsert should succeed"); + + // Only one record should exist + let creds = TuleapCredentials::get(&conn) + .expect("get should succeed") + .expect("should have credentials"); + + assert_eq!(creds.id, second.id); + assert_eq!(creds.tuleap_url, "https://new.example.com"); + assert_eq!(creds.username, "new_user"); + } + + #[test] + fn test_get_returns_none_when_empty() { + let conn = setup(); + let result = TuleapCredentials::get(&conn).expect("get should succeed"); + assert!(result.is_none()); + } + + #[test] + fn test_get_returns_credentials() { + let conn = setup(); + let created = TuleapCredentials::upsert( + &conn, + "https://tuleap.example.com", + "bob", + "enc_pass", + ) + .expect("upsert should succeed"); + + let fetched = TuleapCredentials::get(&conn) + .expect("get should succeed") + .expect("should have credentials"); + + assert_eq!(fetched.id, created.id); + assert_eq!(fetched.tuleap_url, "https://tuleap.example.com"); + assert_eq!(fetched.username, "bob"); + assert_eq!(fetched.password_encrypted, "enc_pass"); + } + + #[test] + fn test_delete_removes_credentials() { + let conn = setup(); + TuleapCredentials::upsert(&conn, "https://tuleap.example.com", "carol", "enc") + .expect("upsert should succeed"); + + TuleapCredentials::delete(&conn).expect("delete should succeed"); + + let result = TuleapCredentials::get(&conn).expect("get should succeed"); + assert!(result.is_none()); + } +} diff --git a/src-tauri/src/models/mod.rs b/src-tauri/src/models/mod.rs new file mode 100644 index 0000000..5caa5db --- /dev/null +++ b/src-tauri/src/models/mod.rs @@ -0,0 +1,4 @@ +pub mod credential; +pub mod project; +pub mod ticket; +pub mod tracker; diff --git a/src-tauri/src/models/project.rs b/src-tauri/src/models/project.rs new file mode 100644 index 0000000..e91dba9 --- /dev/null +++ b/src-tauri/src/models/project.rs @@ -0,0 +1,190 @@ +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Project { + pub id: String, + pub name: String, + pub path: String, + pub cloned_from: Option, + pub base_branch: String, + pub created_at: String, +} + +impl Project { + pub fn insert( + conn: &Connection, + name: &str, + path: &str, + cloned_from: Option<&str>, + base_branch: &str, + ) -> Result { + let id = Uuid::new_v4().to_string(); + let now = chrono::Utc::now().to_rfc3339(); + + conn.execute( + "INSERT INTO projects (id, name, path, cloned_from, base_branch, created_at) VALUES (?1, ?2, ?3, ?4, ?5, ?6)", + params![id, name, path, cloned_from, base_branch, now], + )?; + + Ok(Project { + id, + name: name.to_string(), + path: path.to_string(), + cloned_from: cloned_from.map(String::from), + base_branch: base_branch.to_string(), + created_at: now, + }) + } + + pub fn list(conn: &Connection) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, name, path, cloned_from, base_branch, created_at FROM projects ORDER BY created_at DESC", + )?; + let rows = stmt.query_map([], |row| { + Ok(Project { + id: row.get(0)?, + name: row.get(1)?, + path: row.get(2)?, + cloned_from: row.get(3)?, + base_branch: row.get(4)?, + created_at: row.get(5)?, + }) + })?; + rows.collect() + } + + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + conn.query_row( + "SELECT id, name, path, cloned_from, base_branch, created_at FROM projects WHERE id = ?1", + params![id], + |row| { + Ok(Project { + id: row.get(0)?, + name: row.get(1)?, + path: row.get(2)?, + cloned_from: row.get(3)?, + base_branch: row.get(4)?, + created_at: row.get(5)?, + }) + }, + ) + } + + pub fn update(conn: &Connection, id: &str, name: &str, base_branch: &str) -> Result<()> { + let affected = conn.execute( + "UPDATE projects SET name = ?1, base_branch = ?2 WHERE id = ?3", + params![name, base_branch, id], + )?; + if affected == 0 { + return Err(rusqlite::Error::QueryReturnedNoRows); + } + Ok(()) + } + + pub fn delete(conn: &Connection, id: &str) -> Result<()> { + let affected = conn.execute("DELETE FROM projects WHERE id = ?1", params![id])?; + if affected == 0 { + return Err(rusqlite::Error::QueryReturnedNoRows); + } + Ok(()) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + + fn setup() -> Connection { + db::init_in_memory().expect("db init should succeed") + } + + #[test] + fn test_insert_project_local_path() { + let conn = setup(); + let project = Project::insert(&conn, "My Project", "/home/user/code/myproject", None, "main") + .expect("insert should succeed"); + + assert_eq!(project.name, "My Project"); + assert_eq!(project.path, "/home/user/code/myproject"); + assert!(project.cloned_from.is_none()); + assert_eq!(project.base_branch, "main"); + assert!(!project.id.is_empty()); + assert!(!project.created_at.is_empty()); + } + + #[test] + fn test_insert_project_cloned() { + let conn = setup(); + let project = Project::insert( + &conn, + "Cloned Project", + "/home/user/code/cloned", + Some("https://github.com/org/repo.git"), + "stable", + ) + .expect("insert should succeed"); + + assert_eq!(project.cloned_from.as_deref(), Some("https://github.com/org/repo.git")); + assert_eq!(project.base_branch, "stable"); + } + + #[test] + fn test_list_projects_empty() { + let conn = setup(); + let projects = Project::list(&conn).expect("list should succeed"); + assert!(projects.is_empty()); + } + + #[test] + fn test_list_projects_returns_all() { + let conn = setup(); + Project::insert(&conn, "A", "/path/a", None, "main").unwrap(); + Project::insert(&conn, "B", "/path/b", None, "main").unwrap(); + + let projects = Project::list(&conn).expect("list should succeed"); + assert_eq!(projects.len(), 2); + } + + #[test] + fn test_get_by_id() { + let conn = setup(); + let created = Project::insert(&conn, "Test", "/path/test", None, "main").unwrap(); + let found = Project::get_by_id(&conn, &created.id).expect("get should succeed"); + + assert_eq!(found.id, created.id); + assert_eq!(found.name, "Test"); + } + + #[test] + fn test_get_by_id_not_found() { + let conn = setup(); + let result = Project::get_by_id(&conn, "nonexistent"); + assert!(result.is_err()); + } + + #[test] + fn test_update_project() { + let conn = setup(); + let created = Project::insert(&conn, "Old Name", "/path", None, "main").unwrap(); + + Project::update(&conn, &created.id, "New Name", "develop").expect("update should succeed"); + + let updated = Project::get_by_id(&conn, &created.id).unwrap(); + assert_eq!(updated.name, "New Name"); + assert_eq!(updated.base_branch, "develop"); + } + + #[test] + fn test_delete_project() { + let conn = setup(); + let created = Project::insert(&conn, "ToDelete", "/path", None, "main").unwrap(); + + Project::delete(&conn, &created.id).expect("delete should succeed"); + + let result = Project::get_by_id(&conn, &created.id); + assert!(result.is_err()); + } +} diff --git a/src-tauri/src/models/ticket.rs b/src-tauri/src/models/ticket.rs new file mode 100644 index 0000000..201ce8e --- /dev/null +++ b/src-tauri/src/models/ticket.rs @@ -0,0 +1,242 @@ +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct ProcessedTicket { + pub id: String, + pub tracker_id: String, + pub artifact_id: i32, + pub artifact_title: String, + pub artifact_data: String, + pub status: String, + pub analyst_report: Option, + pub developer_report: Option, + pub worktree_path: Option, + pub branch_name: Option, + pub detected_at: String, + pub processed_at: Option, +} + +fn from_row(row: &rusqlite::Row) -> rusqlite::Result { + Ok(ProcessedTicket { + id: row.get(0)?, + tracker_id: row.get(1)?, + artifact_id: row.get(2)?, + artifact_title: row.get(3)?, + artifact_data: row.get(4)?, + status: row.get(5)?, + analyst_report: row.get(6)?, + developer_report: row.get(7)?, + worktree_path: row.get(8)?, + branch_name: row.get(9)?, + detected_at: row.get(10)?, + processed_at: row.get(11)?, + }) +} + +#[allow(dead_code)] +const SELECT_ALL_COLS: &str = "SELECT id, tracker_id, artifact_id, artifact_title, artifact_data, \ + status, analyst_report, developer_report, worktree_path, branch_name, \ + detected_at, processed_at FROM processed_tickets"; + +impl ProcessedTicket { + /// Insert a new ticket if one with the same (tracker_id, artifact_id) doesn't exist. + /// Returns Some(ticket) if inserted, None if it was a duplicate. + pub fn insert_if_new( + conn: &Connection, + tracker_id: &str, + artifact_id: i32, + artifact_title: &str, + artifact_data: &str, + ) -> Result> { + if Self::exists(conn, tracker_id, artifact_id)? { + return Ok(None); + } + + let id = Uuid::new_v4().to_string(); + let now = chrono::Utc::now().to_rfc3339(); + + conn.execute( + "INSERT INTO processed_tickets \ + (id, tracker_id, artifact_id, artifact_title, artifact_data, status, detected_at) \ + VALUES (?1, ?2, ?3, ?4, ?5, 'Pending', ?6)", + params![id, tracker_id, artifact_id, artifact_title, artifact_data, now], + )?; + + let ticket = ProcessedTicket { + id, + tracker_id: tracker_id.to_string(), + artifact_id, + artifact_title: artifact_title.to_string(), + artifact_data: artifact_data.to_string(), + status: "Pending".to_string(), + analyst_report: None, + developer_report: None, + worktree_path: None, + branch_name: None, + detected_at: now, + processed_at: None, + }; + + Ok(Some(ticket)) + } + + /// Returns true if a ticket with (tracker_id, artifact_id) already exists. + pub fn exists(conn: &Connection, tracker_id: &str, artifact_id: i32) -> Result { + let count: i64 = conn.query_row( + "SELECT COUNT(*) FROM processed_tickets WHERE tracker_id = ?1 AND artifact_id = ?2", + params![tracker_id, artifact_id], + |row| row.get(0), + )?; + Ok(count > 0) + } + + #[allow(dead_code)] + pub fn list_by_tracker(conn: &Connection, tracker_id: &str) -> Result> { + let sql = format!( + "{} WHERE tracker_id = ?1 ORDER BY detected_at DESC", + SELECT_ALL_COLS + ); + let mut stmt = conn.prepare(&sql)?; + let rows = stmt.query_map(params![tracker_id], from_row)?; + rows.collect() + } + + pub fn list_by_project(conn: &Connection, project_id: &str) -> Result> { + let mut stmt = conn.prepare( + "SELECT pt.id, pt.tracker_id, pt.artifact_id, pt.artifact_title, pt.artifact_data, \ + pt.status, pt.analyst_report, pt.developer_report, pt.worktree_path, pt.branch_name, \ + pt.detected_at, pt.processed_at \ + FROM processed_tickets pt \ + JOIN watched_trackers wt ON pt.tracker_id = wt.id \ + WHERE wt.project_id = ?1 \ + ORDER BY pt.detected_at DESC", + )?; + let rows = stmt.query_map(params![project_id], from_row)?; + rows.collect() + } + + #[allow(dead_code)] + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + let sql = format!("{} WHERE id = ?1", SELECT_ALL_COLS); + conn.query_row(&sql, params![id], from_row) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + use crate::models::project::Project; + use crate::models::tracker::{AgentConfig, WatchedTracker}; + + fn setup() -> (Connection, String) { + let conn = db::init_in_memory().expect("db init should succeed"); + let project = Project::insert(&conn, "Test", "/path", None, "main").unwrap(); + let agent_config = AgentConfig { + analyst_command: "claude".into(), + analyst_args: vec![], + developer_command: "claude".into(), + developer_args: vec![], + }; + let tracker = + WatchedTracker::insert(&conn, &project.id, 456, "Bugs", 10, agent_config, vec![]) + .unwrap(); + (conn, tracker.id) + } + + #[test] + fn test_insert_if_new_creates_ticket() { + let (conn, tracker_id) = setup(); + + let result = ProcessedTicket::insert_if_new( + &conn, + &tracker_id, + 101, + "Fix login bug", + "{\"id\": 101}", + ) + .expect("insert_if_new should succeed"); + + assert!(result.is_some()); + let ticket = result.unwrap(); + assert_eq!(ticket.status, "Pending"); + assert_eq!(ticket.tracker_id, tracker_id); + assert_eq!(ticket.artifact_id, 101); + assert_eq!(ticket.artifact_title, "Fix login bug"); + assert!(ticket.analyst_report.is_none()); + assert!(ticket.processed_at.is_none()); + } + + #[test] + fn test_insert_if_new_returns_none_for_duplicate() { + let (conn, tracker_id) = setup(); + + let first = ProcessedTicket::insert_if_new( + &conn, + &tracker_id, + 202, + "Crash on startup", + "{\"id\": 202}", + ) + .expect("first insert should succeed"); + assert!(first.is_some()); + + let second = ProcessedTicket::insert_if_new( + &conn, + &tracker_id, + 202, + "Crash on startup", + "{\"id\": 202}", + ) + .expect("second insert_if_new should succeed"); + assert!(second.is_none()); + } + + #[test] + fn test_exists() { + let (conn, tracker_id) = setup(); + + let before = ProcessedTicket::exists(&conn, &tracker_id, 303) + .expect("exists check should succeed"); + assert!(!before); + + ProcessedTicket::insert_if_new(&conn, &tracker_id, 303, "Some ticket", "{}") + .expect("insert should succeed"); + + let after = ProcessedTicket::exists(&conn, &tracker_id, 303) + .expect("exists check after insert should succeed"); + assert!(after); + } + + #[test] + fn test_list_by_tracker() { + let (conn, tracker_id) = setup(); + + ProcessedTicket::insert_if_new(&conn, &tracker_id, 1, "Ticket One", "{}").unwrap(); + ProcessedTicket::insert_if_new(&conn, &tracker_id, 2, "Ticket Two", "{}").unwrap(); + + let tickets = + ProcessedTicket::list_by_tracker(&conn, &tracker_id).expect("list should succeed"); + assert_eq!(tickets.len(), 2); + } + + #[test] + fn test_get_by_id() { + let (conn, tracker_id) = setup(); + + let inserted = + ProcessedTicket::insert_if_new(&conn, &tracker_id, 404, "Not Found Bug", "{\"id\": 404}") + .expect("insert should succeed") + .expect("should be Some"); + + let found = + ProcessedTicket::get_by_id(&conn, &inserted.id).expect("get_by_id should succeed"); + + assert_eq!(found.id, inserted.id); + assert_eq!(found.artifact_id, 404); + assert_eq!(found.artifact_title, "Not Found Bug"); + assert_eq!(found.status, "Pending"); + } +} diff --git a/src-tauri/src/models/tracker.rs b/src-tauri/src/models/tracker.rs new file mode 100644 index 0000000..252df15 --- /dev/null +++ b/src-tauri/src/models/tracker.rs @@ -0,0 +1,359 @@ +use rusqlite::{params, Connection, Result}; +use serde::{Deserialize, Serialize}; +use uuid::Uuid; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct AgentConfig { + pub analyst_command: String, + pub analyst_args: Vec, + pub developer_command: String, + pub developer_args: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct FilterGroup { + pub conditions: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Filter { + pub field: String, + pub operator: String, // "In", "NotIn", "Equals", "NotEquals" + pub value: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct WatchedTracker { + pub id: String, + pub project_id: String, + pub tracker_id: i32, + pub tracker_label: String, + pub polling_interval: i32, + pub agent_config: AgentConfig, + pub filters: Vec, + pub enabled: bool, + pub last_polled_at: Option, + pub created_at: String, +} + +fn from_row(row: &rusqlite::Row) -> rusqlite::Result { + let agent_config_json: String = row.get(5)?; + let filters_json: String = row.get(6)?; + let enabled_int: i32 = row.get(7)?; + + let agent_config: AgentConfig = serde_json::from_str(&agent_config_json) + .map_err(|e| rusqlite::Error::FromSqlConversionFailure(5, rusqlite::types::Type::Text, Box::new(e)))?; + let filters: Vec = serde_json::from_str(&filters_json) + .map_err(|e| rusqlite::Error::FromSqlConversionFailure(6, rusqlite::types::Type::Text, Box::new(e)))?; + + Ok(WatchedTracker { + id: row.get(0)?, + project_id: row.get(1)?, + tracker_id: row.get(2)?, + tracker_label: row.get(3)?, + polling_interval: row.get(4)?, + agent_config, + filters, + enabled: enabled_int != 0, + last_polled_at: row.get(8)?, + created_at: row.get(9)?, + }) +} + +impl WatchedTracker { + pub fn insert( + conn: &Connection, + project_id: &str, + tracker_id: i32, + tracker_label: &str, + polling_interval: i32, + agent_config: AgentConfig, + filters: Vec, + ) -> Result { + let id = Uuid::new_v4().to_string(); + let now = chrono::Utc::now().to_rfc3339(); + + let agent_config_json = serde_json::to_string(&agent_config) + .map_err(|e| rusqlite::Error::ToSqlConversionFailure(Box::new(e)))?; + let filters_json = serde_json::to_string(&filters) + .map_err(|e| rusqlite::Error::ToSqlConversionFailure(Box::new(e)))?; + + conn.execute( + "INSERT INTO watched_trackers (id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, created_at) \ + VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8)", + params![id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, now], + )?; + + Ok(WatchedTracker { + id, + project_id: project_id.to_string(), + tracker_id, + tracker_label: tracker_label.to_string(), + polling_interval, + agent_config, + filters, + enabled: true, + last_polled_at: None, + created_at: now, + }) + } + + pub fn list_by_project(conn: &Connection, project_id: &str) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, enabled, last_polled_at, created_at \ + FROM watched_trackers WHERE project_id = ?1 ORDER BY created_at DESC", + )?; + let rows = stmt.query_map(params![project_id], from_row)?; + rows.collect() + } + + pub fn list_all_enabled(conn: &Connection) -> Result> { + let mut stmt = conn.prepare( + "SELECT id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, enabled, last_polled_at, created_at \ + FROM watched_trackers WHERE enabled = 1 ORDER BY created_at DESC", + )?; + let rows = stmt.query_map([], from_row)?; + rows.collect() + } + + pub fn get_by_id(conn: &Connection, id: &str) -> Result { + conn.query_row( + "SELECT id, project_id, tracker_id, tracker_label, polling_interval, agent_config_json, filters_json, enabled, last_polled_at, created_at \ + FROM watched_trackers WHERE id = ?1", + params![id], + from_row, + ) + } + + pub fn update( + conn: &Connection, + id: &str, + polling_interval: i32, + agent_config: AgentConfig, + filters: Vec, + enabled: bool, + ) -> Result<()> { + let agent_config_json = serde_json::to_string(&agent_config) + .map_err(|e| rusqlite::Error::ToSqlConversionFailure(Box::new(e)))?; + let filters_json = serde_json::to_string(&filters) + .map_err(|e| rusqlite::Error::ToSqlConversionFailure(Box::new(e)))?; + let enabled_int = if enabled { 1i32 } else { 0i32 }; + + let affected = conn.execute( + "UPDATE watched_trackers SET polling_interval = ?1, agent_config_json = ?2, filters_json = ?3, enabled = ?4 WHERE id = ?5", + params![polling_interval, agent_config_json, filters_json, enabled_int, id], + )?; + + if affected == 0 { + return Err(rusqlite::Error::QueryReturnedNoRows); + } + Ok(()) + } + + pub fn update_last_polled(conn: &Connection, id: &str) -> Result<()> { + let now = chrono::Utc::now().to_rfc3339(); + let affected = conn.execute( + "UPDATE watched_trackers SET last_polled_at = ?1 WHERE id = ?2", + params![now, id], + )?; + if affected == 0 { + return Err(rusqlite::Error::QueryReturnedNoRows); + } + Ok(()) + } + + pub fn delete(conn: &Connection, id: &str) -> Result<()> { + let affected = conn.execute("DELETE FROM watched_trackers WHERE id = ?1", params![id])?; + if affected == 0 { + return Err(rusqlite::Error::QueryReturnedNoRows); + } + Ok(()) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::db; + use crate::models::project::Project; + + fn setup() -> Connection { + let conn = db::init_in_memory().expect("db init should succeed"); + Project::insert(&conn, "Test Project", "/path/test", None, "main").unwrap(); + conn + } + + fn project_id(conn: &Connection) -> String { + Project::list(conn).unwrap().into_iter().next().unwrap().id + } + + fn sample_agent_config() -> AgentConfig { + AgentConfig { + analyst_command: "analyst".to_string(), + analyst_args: vec!["--mode".to_string(), "analyze".to_string()], + developer_command: "developer".to_string(), + developer_args: vec!["--fix".to_string()], + } + } + + fn sample_filters() -> Vec { + vec![FilterGroup { + conditions: vec![Filter { + field: "status".to_string(), + operator: "In".to_string(), + value: vec!["Open".to_string(), "In Progress".to_string()], + }], + }] + } + + #[test] + fn test_insert_tracker() { + let conn = setup(); + let pid = project_id(&conn); + + let tracker = WatchedTracker::insert( + &conn, + &pid, + 42, + "Bug Tracker", + 15, + sample_agent_config(), + sample_filters(), + ) + .expect("insert should succeed"); + + assert!(!tracker.id.is_empty()); + assert_eq!(tracker.project_id, pid); + assert_eq!(tracker.tracker_id, 42); + assert_eq!(tracker.tracker_label, "Bug Tracker"); + assert_eq!(tracker.polling_interval, 15); + assert!(tracker.enabled); + assert!(tracker.last_polled_at.is_none()); + assert!(!tracker.created_at.is_empty()); + assert_eq!(tracker.agent_config.analyst_command, "analyst"); + assert_eq!(tracker.filters.len(), 1); + } + + #[test] + fn test_list_by_project() { + let conn = setup(); + let pid = project_id(&conn); + + WatchedTracker::insert(&conn, &pid, 1, "Tracker A", 10, sample_agent_config(), vec![]).unwrap(); + WatchedTracker::insert(&conn, &pid, 2, "Tracker B", 20, sample_agent_config(), vec![]).unwrap(); + + let trackers = WatchedTracker::list_by_project(&conn, &pid).expect("list should succeed"); + assert_eq!(trackers.len(), 2); + } + + #[test] + fn test_list_all_enabled() { + let conn = setup(); + let pid = project_id(&conn); + + let t1 = WatchedTracker::insert(&conn, &pid, 1, "Enabled", 10, sample_agent_config(), vec![]).unwrap(); + let t2 = WatchedTracker::insert(&conn, &pid, 2, "Disabled", 10, sample_agent_config(), vec![]).unwrap(); + + // Disable t2 + WatchedTracker::update( + &conn, + &t2.id, + t2.polling_interval, + sample_agent_config(), + vec![], + false, + ) + .unwrap(); + + let enabled = WatchedTracker::list_all_enabled(&conn).expect("list_all_enabled should succeed"); + assert_eq!(enabled.len(), 1); + assert_eq!(enabled[0].id, t1.id); + } + + #[test] + fn test_get_by_id() { + let conn = setup(); + let pid = project_id(&conn); + + let created = WatchedTracker::insert( + &conn, + &pid, + 99, + "My Tracker", + 30, + sample_agent_config(), + sample_filters(), + ) + .unwrap(); + + let found = WatchedTracker::get_by_id(&conn, &created.id).expect("get_by_id should succeed"); + assert_eq!(found.id, created.id); + assert_eq!(found.tracker_id, 99); + assert_eq!(found.tracker_label, "My Tracker"); + assert_eq!(found.polling_interval, 30); + assert_eq!(found.filters.len(), 1); + } + + #[test] + fn test_update_tracker() { + let conn = setup(); + let pid = project_id(&conn); + + let created = WatchedTracker::insert( + &conn, + &pid, + 10, + "Original", + 5, + sample_agent_config(), + sample_filters(), + ) + .unwrap(); + + let new_filters = vec![FilterGroup { + conditions: vec![Filter { + field: "priority".to_string(), + operator: "Equals".to_string(), + value: vec!["High".to_string()], + }], + }]; + + WatchedTracker::update(&conn, &created.id, 60, sample_agent_config(), new_filters, false) + .expect("update should succeed"); + + let updated = WatchedTracker::get_by_id(&conn, &created.id).unwrap(); + assert_eq!(updated.polling_interval, 60); + assert!(!updated.enabled); + assert_eq!(updated.filters[0].conditions[0].field, "priority"); + } + + #[test] + fn test_update_last_polled() { + let conn = setup(); + let pid = project_id(&conn); + + let created = + WatchedTracker::insert(&conn, &pid, 5, "Poller", 10, sample_agent_config(), vec![]).unwrap(); + + assert!(created.last_polled_at.is_none()); + + WatchedTracker::update_last_polled(&conn, &created.id).expect("update_last_polled should succeed"); + + let updated = WatchedTracker::get_by_id(&conn, &created.id).unwrap(); + assert!(updated.last_polled_at.is_some()); + } + + #[test] + fn test_delete_tracker() { + let conn = setup(); + let pid = project_id(&conn); + + let created = + WatchedTracker::insert(&conn, &pid, 7, "ToDelete", 10, sample_agent_config(), vec![]).unwrap(); + + WatchedTracker::delete(&conn, &created.id).expect("delete should succeed"); + + let result = WatchedTracker::get_by_id(&conn, &created.id); + assert!(result.is_err()); + } +} diff --git a/src-tauri/src/services/crypto.rs b/src-tauri/src/services/crypto.rs new file mode 100644 index 0000000..979547d --- /dev/null +++ b/src-tauri/src/services/crypto.rs @@ -0,0 +1,105 @@ +use aes_gcm::{ + aead::{Aead, KeyInit, OsRng}, + Aes256Gcm, Key, Nonce, +}; +use base64::{engine::general_purpose::STANDARD, Engine}; +use rand::RngCore; + +pub fn encrypt(key: &[u8; 32], plaintext: &str) -> Result { + let cipher = Aes256Gcm::new(Key::::from_slice(key)); + let mut nonce_bytes = [0u8; 12]; + OsRng.fill_bytes(&mut nonce_bytes); + let nonce = Nonce::from_slice(&nonce_bytes); + let ciphertext = cipher + .encrypt(nonce, plaintext.as_bytes()) + .map_err(|e| format!("encryption failed: {}", e))?; + let mut combined = nonce_bytes.to_vec(); + combined.extend(ciphertext); + Ok(STANDARD.encode(&combined)) +} + +pub fn decrypt(key: &[u8; 32], encrypted: &str) -> Result { + let combined = STANDARD + .decode(encrypted) + .map_err(|e| format!("base64 decode failed: {}", e))?; + if combined.len() < 13 { + return Err("encrypted data too short".to_string()); + } + let (nonce_bytes, ciphertext) = combined.split_at(12); + let cipher = Aes256Gcm::new(Key::::from_slice(key)); + let nonce = Nonce::from_slice(nonce_bytes); + let plaintext = cipher + .decrypt(nonce, ciphertext) + .map_err(|_| "decryption failed (wrong key or corrupted data)".to_string())?; + String::from_utf8(plaintext).map_err(|e| format!("invalid UTF-8: {}", e)) +} + +#[cfg(test)] +mod tests { + use super::*; + + fn test_key() -> [u8; 32] { + [42u8; 32] + } + + #[test] + fn test_encrypt_decrypt_roundtrip() { + let key = test_key(); + let plaintext = "hello world"; + let encrypted = encrypt(&key, plaintext).unwrap(); + let decrypted = decrypt(&key, &encrypted).unwrap(); + assert_eq!(decrypted, plaintext); + } + + #[test] + fn test_encrypt_produces_different_ciphertext() { + let key = test_key(); + let plaintext = "same input"; + let enc1 = encrypt(&key, plaintext).unwrap(); + let enc2 = encrypt(&key, plaintext).unwrap(); + assert_ne!(enc1, enc2); + } + + #[test] + fn test_decrypt_with_wrong_key_fails() { + let key = test_key(); + let wrong_key = [99u8; 32]; + let encrypted = encrypt(&key, "secret").unwrap(); + let result = decrypt(&wrong_key, &encrypted); + assert!(result.is_err()); + } + + #[test] + fn test_decrypt_invalid_base64_fails() { + let key = test_key(); + let result = decrypt(&key, "not valid base64!!!"); + assert!(result.is_err()); + } + + #[test] + fn test_decrypt_too_short_fails() { + let key = test_key(); + // Base64 of 5 bytes (less than 13) + let short = STANDARD.encode(&[0u8; 5]); + let result = decrypt(&key, &short); + assert!(result.is_err()); + assert!(result.unwrap_err().contains("too short")); + } + + #[test] + fn test_encrypt_empty_string() { + let key = test_key(); + let encrypted = encrypt(&key, "").unwrap(); + let decrypted = decrypt(&key, &encrypted).unwrap(); + assert_eq!(decrypted, ""); + } + + #[test] + fn test_encrypt_unicode() { + let key = test_key(); + let plaintext = "héllo wörld àèìòù"; + let encrypted = encrypt(&key, plaintext).unwrap(); + let decrypted = decrypt(&key, &encrypted).unwrap(); + assert_eq!(decrypted, plaintext); + } +} diff --git a/src-tauri/src/services/filter_engine.rs b/src-tauri/src/services/filter_engine.rs new file mode 100644 index 0000000..2334dcf --- /dev/null +++ b/src-tauri/src/services/filter_engine.rs @@ -0,0 +1,212 @@ +use crate::models::tracker::{Filter, FilterGroup}; +use crate::services::tuleap_client::extract_artifact_field_values; + +pub fn apply_filters( + artifacts: &[serde_json::Value], + filter_groups: &[FilterGroup], +) -> Vec { + if filter_groups.is_empty() { + return artifacts.to_vec(); + } + artifacts + .iter() + .filter(|a| matches_all_groups(a, filter_groups)) + .cloned() + .collect() +} + +fn matches_all_groups(artifact: &serde_json::Value, groups: &[FilterGroup]) -> bool { + groups.iter().all(|g| matches_any_condition(artifact, &g.conditions)) +} + +fn matches_any_condition(artifact: &serde_json::Value, conditions: &[Filter]) -> bool { + if conditions.is_empty() { + return true; + } + conditions.iter().any(|c| matches_condition(artifact, c)) +} + +fn matches_condition(artifact: &serde_json::Value, condition: &Filter) -> bool { + let field_values = extract_artifact_field_values(artifact, &condition.field); + match condition.operator.as_str() { + "Equals" => { + condition.value.len() == 1 + && field_values.iter().any(|v| v == &condition.value[0]) + } + "NotEquals" => { + condition.value.len() == 1 + && !field_values.iter().any(|v| v == &condition.value[0]) + } + "In" => field_values.iter().any(|v| condition.value.contains(v)), + "NotIn" => !field_values.iter().any(|v| condition.value.contains(v)), + _ => false, + } +} + +#[cfg(test)] +mod tests { + use super::*; + use serde_json::json; + + fn make_artifact(status: &str, assigned: &str, priority: &str) -> serde_json::Value { + json!({ + "id": 123, + "title": "Test ticket", + "values": [ + { + "field_id": 1, + "label": "Status", + "type": "sb", + "values": [{ "id": 1, "label": status }] + }, + { + "field_id": 2, + "label": "Assigned to", + "type": "msb", + "bind_value_objects": [{ "id": 2, "display_name": assigned }] + }, + { + "field_id": 3, + "label": "Priority", + "type": "sb", + "values": [{ "id": 3, "label": priority }] + } + ] + }) + } + + #[test] + fn test_empty_filters_returns_all() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Ferme", "Bob", "Basse"), + ]; + let result = apply_filters(&artifacts, &[]); + assert_eq!(result.len(), 2); + } + + #[test] + fn test_single_in_filter() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Ferme", "Bob", "Basse"), + ]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "In".to_string(), + value: vec!["Nouveau".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 1); + assert_eq!(result[0]["values"][0]["values"][0]["label"], "Nouveau"); + } + + #[test] + fn test_or_within_group() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("A traiter", "Bob", "Moyenne"), + make_artifact("Ferme", "Carol", "Basse"), + ]; + let groups = vec![FilterGroup { + conditions: vec![ + Filter { + field: "Status".to_string(), + operator: "Equals".to_string(), + value: vec!["Nouveau".to_string()], + }, + Filter { + field: "Status".to_string(), + operator: "Equals".to_string(), + value: vec!["A traiter".to_string()], + }, + ], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 2); + } + + #[test] + fn test_and_across_groups() { + let artifacts = vec![ + make_artifact("Nouveau", "Team Maintenance", "Haute"), + make_artifact("A traiter", "Other Team", "Moyenne"), + make_artifact("Ferme", "Team Maintenance", "Basse"), + ]; + let groups = vec![ + FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "In".to_string(), + value: vec!["Nouveau".to_string(), "A traiter".to_string()], + }], + }, + FilterGroup { + conditions: vec![Filter { + field: "Assigned to".to_string(), + operator: "In".to_string(), + value: vec!["Team Maintenance".to_string()], + }], + }, + ]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 1); + assert_eq!(result[0]["values"][0]["values"][0]["label"], "Nouveau"); + } + + #[test] + fn test_not_in_filter() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Ferme", "Bob", "Basse"), + make_artifact("Ferme", "Carol", "Moyenne"), + ]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "NotIn".to_string(), + value: vec!["Ferme".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 1); + assert_eq!(result[0]["values"][0]["values"][0]["label"], "Nouveau"); + } + + #[test] + fn test_equals_filter() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("Nouveau", "Bob", "Basse"), + make_artifact("Ferme", "Carol", "Haute"), + ]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Priority".to_string(), + operator: "Equals".to_string(), + value: vec!["Haute".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 2); + } + + #[test] + fn test_no_match_returns_empty() { + let artifacts = vec![ + make_artifact("Nouveau", "Alice", "Haute"), + make_artifact("A traiter", "Bob", "Moyenne"), + ]; + let groups = vec![FilterGroup { + conditions: vec![Filter { + field: "Status".to_string(), + operator: "Equals".to_string(), + value: vec!["Ferme".to_string()], + }], + }]; + let result = apply_filters(&artifacts, &groups); + assert_eq!(result.len(), 0); + } +} diff --git a/src-tauri/src/services/mod.rs b/src-tauri/src/services/mod.rs new file mode 100644 index 0000000..c335479 --- /dev/null +++ b/src-tauri/src/services/mod.rs @@ -0,0 +1,4 @@ +pub mod crypto; +pub mod filter_engine; +pub mod poller; +pub mod tuleap_client; diff --git a/src-tauri/src/services/poller.rs b/src-tauri/src/services/poller.rs new file mode 100644 index 0000000..66c0c55 --- /dev/null +++ b/src-tauri/src/services/poller.rs @@ -0,0 +1,180 @@ +use crate::models::credential::TuleapCredentials; +use crate::models::ticket::ProcessedTicket; +use crate::models::tracker::WatchedTracker; +use crate::services::{crypto, filter_engine}; +use crate::services::tuleap_client::TuleapClient; +use rusqlite::Connection; +use std::sync::{Arc, Mutex}; +use tauri::{AppHandle, Emitter}; +use tokio::time::{interval, Duration}; + +pub fn start( + db: Arc>, + encryption_key: [u8; 32], + http_client: reqwest::Client, + app_handle: AppHandle, +) { + tokio::spawn(async move { + let mut tick = interval(Duration::from_secs(60)); + loop { + tick.tick().await; + poll_all_trackers(&db, &encryption_key, &http_client, &app_handle).await; + } + }); +} + +async fn poll_all_trackers( + db: &Arc>, + encryption_key: &[u8; 32], + http_client: &reqwest::Client, + app_handle: &AppHandle, +) { + // 1. Read all enabled trackers and credentials from DB + let (trackers, client) = { + let conn = match db.lock() { + Ok(c) => c, + Err(e) => { + eprintln!("poller: failed to lock db: {}", e); + return; + } + }; + + let trackers = match WatchedTracker::list_all_enabled(&conn) { + Ok(t) => t, + Err(e) => { + eprintln!("poller: failed to list trackers: {}", e); + return; + } + }; + + // 2. Read credentials; bail silently if none + let creds = match TuleapCredentials::get(&conn) { + Ok(Some(c)) => c, + Ok(None) => return, + Err(e) => { + eprintln!("poller: failed to read credentials: {}", e); + return; + } + }; + + let password = match crypto::decrypt(encryption_key, &creds.password_encrypted) { + Ok(p) => p, + Err(e) => { + eprintln!("poller: failed to decrypt password: {}", e); + return; + } + }; + + let client = TuleapClient::new(http_client, &creds.tuleap_url, &creds.username, &password); + + (trackers, client) + }; // lock released + + // 3. For each tracker that should_poll, poll it + for tracker in &trackers { + if should_poll(tracker) { + poll_single_tracker(db, &client, tracker, app_handle).await; + } + } +} + +fn should_poll(tracker: &WatchedTracker) -> bool { + let last_polled_at = match &tracker.last_polled_at { + None => return true, // Never polled + Some(s) => s, + }; + + let last = match chrono::DateTime::parse_from_rfc3339(last_polled_at) { + Ok(dt) => dt, + Err(e) => { + eprintln!("poller: failed to parse last_polled_at '{}': {}", last_polled_at, e); + return true; // Treat as never polled on parse error + } + }; + + let elapsed = chrono::Utc::now().signed_duration_since(last).num_minutes(); + elapsed >= tracker.polling_interval as i64 +} + +async fn poll_single_tracker( + db: &Arc>, + client: &TuleapClient, + tracker: &WatchedTracker, + app_handle: &AppHandle, +) { + // 1. Fetch artifacts + let artifacts = match client.get_artifacts(tracker.tracker_id).await { + Ok(a) => a, + Err(e) => { + eprintln!("poller: failed to fetch artifacts for tracker {}: {}", tracker.id, e); + return; + } + }; + + // 2. Apply filters + let filtered = filter_engine::apply_filters(&artifacts, &tracker.filters); + + // 3. Insert new tickets and update last_polled_at + let new_count = { + let conn = match db.lock() { + Ok(c) => c, + Err(e) => { + eprintln!("poller: failed to lock db for insert: {}", e); + return; + } + }; + + let mut count = 0usize; + + for artifact in &filtered { + let artifact_id = artifact + .get("id") + .and_then(|v| v.as_i64()) + .unwrap_or(0) as i32; + + let artifact_title = artifact + .get("title") + .and_then(|v| v.as_str()) + .unwrap_or("") + .to_string(); + + let artifact_data = serde_json::to_string(artifact) + .unwrap_or_else(|_| "{}".to_string()); + + match ProcessedTicket::insert_if_new( + &conn, + &tracker.id, + artifact_id, + &artifact_title, + &artifact_data, + ) { + Ok(Some(_)) => count += 1, + Ok(None) => {} + Err(e) => { + eprintln!("poller: failed to insert ticket (artifact {}): {}", artifact_id, e); + } + } + } + + // 4. Update last_polled_at + if let Err(e) = WatchedTracker::update_last_polled(&conn, &tracker.id) { + eprintln!("poller: failed to update last_polled_at for tracker {}: {}", tracker.id, e); + } + + count + }; // lock released + + // 5. Emit event if new tickets found + if new_count > 0 { + if let Err(e) = app_handle.emit( + "new-tickets-detected", + serde_json::json!({ + "tracker_id": tracker.id, + "tracker_label": tracker.tracker_label, + "count": new_count, + }), + ) { + eprintln!("poller: failed to emit event: {}", e); + } + } +} diff --git a/src-tauri/src/services/tuleap_client.rs b/src-tauri/src/services/tuleap_client.rs new file mode 100644 index 0000000..358001f --- /dev/null +++ b/src-tauri/src/services/tuleap_client.rs @@ -0,0 +1,411 @@ +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct TrackerField { + pub field_id: i64, + pub label: String, + pub field_type: String, + pub values: Vec, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct FieldValue { + pub id: i64, + pub label: String, +} + +pub struct TuleapClient { + http: reqwest::Client, + base_url: String, + username: String, + password: String, +} + +impl TuleapClient { + pub fn new(http: &reqwest::Client, base_url: &str, username: &str, password: &str) -> Self { + Self { + http: http.clone(), + base_url: base_url.trim_end_matches('/').to_string(), + username: username.to_string(), + password: password.to_string(), + } + } + + #[allow(dead_code)] + pub async fn test_connection(&self) -> Result<(), String> { + let url = format!("{}/api/projects?limit=1", self.base_url); + let resp = self + .http + .get(&url) + .basic_auth(&self.username, Some(&self.password)) + .send() + .await + .map_err(|e| format!("request failed: {}", e))?; + + if resp.status().is_success() { + Ok(()) + } else { + Err(format!("connection test failed: HTTP {}", resp.status())) + } + } + + pub async fn get_tracker_fields(&self, tracker_id: i32) -> Result, String> { + let url = format!("{}/api/trackers/{}", self.base_url, tracker_id); + let resp = self + .http + .get(&url) + .basic_auth(&self.username, Some(&self.password)) + .send() + .await + .map_err(|e| format!("request failed: {}", e))?; + + if !resp.status().is_success() { + return Err(format!("get tracker fields failed: HTTP {}", resp.status())); + } + + let body: serde_json::Value = resp + .json() + .await + .map_err(|e| format!("failed to parse response: {}", e))?; + + Ok(parse_tracker_fields(&body)) + } + + pub async fn get_artifacts(&self, tracker_id: i32) -> Result, String> { + let mut all_artifacts: Vec = Vec::new(); + let mut offset = 0usize; + let limit = 50usize; + + loop { + let url = format!( + "{}/api/trackers/{}/artifacts?values=all&limit={}&offset={}", + self.base_url, tracker_id, limit, offset + ); + let resp = self + .http + .get(&url) + .basic_auth(&self.username, Some(&self.password)) + .send() + .await + .map_err(|e| format!("request failed: {}", e))?; + + if !resp.status().is_success() { + return Err(format!("get artifacts failed: HTTP {}", resp.status())); + } + + // Read total size from header before consuming body + let total: usize = resp + .headers() + .get("x-pagination-size") + .and_then(|v| v.to_str().ok()) + .and_then(|s| s.parse().ok()) + .unwrap_or(0); + + let body: serde_json::Value = resp + .json() + .await + .map_err(|e| format!("failed to parse response: {}", e))?; + + let artifacts = body + .as_array() + .cloned() + .unwrap_or_default(); + + let fetched = artifacts.len(); + all_artifacts.extend(artifacts); + + offset += fetched; + + if fetched == 0 || offset >= total { + break; + } + } + + Ok(all_artifacts) + } +} + +// --------------------------------------------------------------------------- +// Pure parsing functions (unit-testable without HTTP) +// --------------------------------------------------------------------------- + +pub fn parse_tracker_fields(tracker_json: &serde_json::Value) -> Vec { + let fields = match tracker_json.get("fields").and_then(|f| f.as_array()) { + Some(f) => f, + None => return vec![], + }; + + fields + .iter() + .filter_map(|field| { + let field_id = field.get("field_id")?.as_i64()?; + let label = field.get("label")?.as_str()?.to_string(); + let field_type = field.get("type")?.as_str()?.to_string(); + + let values = match field_type.as_str() { + "sb" | "msb" | "rb" | "cb" => extract_field_values(field), + _ => vec![], + }; + + Some(TrackerField { + field_id, + label, + field_type, + values, + }) + }) + .collect() +} + +fn extract_field_values(field: &serde_json::Value) -> Vec { + // Try "values" first (sb, rb), then "bind_value_objects" (msb) + let candidates = field + .get("values") + .and_then(|v| v.as_array()) + .filter(|arr| !arr.is_empty()) + .or_else(|| { + field + .get("bind_value_objects") + .and_then(|v| v.as_array()) + }); + + let arr = match candidates { + Some(a) => a, + None => return vec![], + }; + + arr.iter() + .filter_map(|v| { + let id = v.get("id")?.as_i64()?; + let label = v + .get("label") + .and_then(|l| l.as_str()) + .unwrap_or("") + .to_string(); + // Filter out "None" labels + if label == "None" { + return None; + } + Some(FieldValue { id, label }) + }) + .collect() +} + +pub fn extract_artifact_field_values( + artifact: &serde_json::Value, + field_label: &str, +) -> Vec { + let values_arr = match artifact.get("values").and_then(|v| v.as_array()) { + Some(a) => a, + None => return vec![], + }; + + // Find the field entry matching the label + let field_entry = values_arr + .iter() + .find(|entry| entry.get("label").and_then(|l| l.as_str()) == Some(field_label)); + + let entry = match field_entry { + Some(e) => e, + None => return vec![], + }; + + let field_type = entry + .get("type") + .and_then(|t| t.as_str()) + .unwrap_or(""); + + match field_type { + "sb" | "rb" => { + // values[*].label + entry + .get("values") + .and_then(|v| v.as_array()) + .map(|arr| { + arr.iter() + .filter_map(|v| v.get("label").and_then(|l| l.as_str()).map(str::to_string)) + .collect() + }) + .unwrap_or_default() + } + "msb" | "cb" => { + // bind_value_objects[*].display_name, fallback to label + entry + .get("bind_value_objects") + .and_then(|v| v.as_array()) + .map(|arr| { + arr.iter() + .filter_map(|v| { + v.get("display_name") + .and_then(|d| d.as_str()) + .or_else(|| v.get("label").and_then(|l| l.as_str())) + .map(str::to_string) + }) + .collect() + }) + .unwrap_or_default() + } + "string" | "text" | "int" | "float" => { + // scalar "value" field + entry + .get("value") + .map(|v| match v { + serde_json::Value::String(s) => vec![s.clone()], + serde_json::Value::Number(n) => vec![n.to_string()], + serde_json::Value::Bool(b) => vec![b.to_string()], + serde_json::Value::Null => vec![], + other => vec![other.to_string()], + }) + .unwrap_or_default() + } + _ => vec![], + } +} + +// --------------------------------------------------------------------------- +// Tests +// --------------------------------------------------------------------------- + +#[cfg(test)] +mod tests { + use super::*; + use serde_json::json; + + #[test] + fn test_parse_tracker_fields_extracts_sb() { + let tracker = json!({ + "fields": [ + { + "field_id": 1, + "label": "Status", + "type": "sb", + "values": [ + { "id": 0, "label": "None" }, + { "id": 1, "label": "Nouveau" }, + { "id": 2, "label": "En cours" } + ] + } + ] + }); + + let fields = parse_tracker_fields(&tracker); + assert_eq!(fields.len(), 1); + let f = &fields[0]; + assert_eq!(f.label, "Status"); + assert_eq!(f.field_type, "sb"); + // "None" is filtered out + assert_eq!(f.values.len(), 2); + assert_eq!(f.values[0].label, "Nouveau"); + assert_eq!(f.values[1].label, "En cours"); + } + + #[test] + fn test_parse_tracker_fields_extracts_msb() { + let tracker = json!({ + "fields": [ + { + "field_id": 2, + "label": "Assigned to", + "type": "msb", + "bind_value_objects": [ + { "id": 10, "label": "Alice" }, + { "id": 11, "label": "Bob" } + ] + } + ] + }); + + let fields = parse_tracker_fields(&tracker); + assert_eq!(fields.len(), 1); + let f = &fields[0]; + assert_eq!(f.label, "Assigned to"); + assert_eq!(f.values.len(), 2); + assert_eq!(f.values[0].label, "Alice"); + assert_eq!(f.values[1].label, "Bob"); + } + + #[test] + fn test_parse_tracker_fields_skips_text_fields() { + let tracker = json!({ + "fields": [ + { + "field_id": 3, + "label": "Summary", + "type": "text" + } + ] + }); + + let fields = parse_tracker_fields(&tracker); + assert_eq!(fields.len(), 1); + let f = &fields[0]; + assert_eq!(f.label, "Summary"); + assert_eq!(f.field_type, "text"); + assert!(f.values.is_empty()); + } + + #[test] + fn test_extract_artifact_field_values_sb() { + let artifact = json!({ + "values": [ + { + "field_id": 1, + "label": "Status", + "type": "sb", + "values": [ + { "id": 1, "label": "Nouveau" } + ] + } + ] + }); + + let result = extract_artifact_field_values(&artifact, "Status"); + assert_eq!(result, vec!["Nouveau"]); + } + + #[test] + fn test_extract_artifact_field_values_msb() { + let artifact = json!({ + "values": [ + { + "field_id": 2, + "label": "Assigned to", + "type": "msb", + "bind_value_objects": [ + { "id": 10, "display_name": "Team Maintenance" } + ] + } + ] + }); + + let result = extract_artifact_field_values(&artifact, "Assigned to"); + assert_eq!(result, vec!["Team Maintenance"]); + } + + #[test] + fn test_extract_artifact_field_values_missing_field() { + let artifact = json!({ + "values": [] + }); + + let result = extract_artifact_field_values(&artifact, "Status"); + assert!(result.is_empty()); + } + + #[test] + fn test_extract_artifact_field_values_string_field() { + let artifact = json!({ + "values": [ + { + "field_id": 5, + "label": "Summary", + "type": "string", + "value": "Login broken" + } + ] + }); + + let result = extract_artifact_field_values(&artifact, "Summary"); + assert_eq!(result, vec!["Login broken"]); + } +} diff --git a/src-tauri/tauri.conf.json b/src-tauri/tauri.conf.json new file mode 100644 index 0000000..aa68e57 --- /dev/null +++ b/src-tauri/tauri.conf.json @@ -0,0 +1,35 @@ +{ + "$schema": "https://schema.tauri.app/config/2", + "productName": "Orchai", + "version": "0.1.0", + "identifier": "com.orchai.app", + "build": { + "beforeDevCommand": "npm run dev", + "devUrl": "http://localhost:1420", + "beforeBuildCommand": "npm run build", + "frontendDist": "../dist" + }, + "app": { + "windows": [ + { + "title": "Orchai", + "width": 1200, + "height": 800 + } + ], + "security": { + "csp": "default-src 'self'; script-src 'self'; style-src 'self' 'unsafe-inline'; connect-src ipc: http://ipc.localhost" + } + }, + "bundle": { + "active": true, + "targets": "all", + "icon": [ + "icons/32x32.png", + "icons/128x128.png", + "icons/128x128@2x.png", + "icons/icon.icns", + "icons/icon.ico" + ] + } +} diff --git a/src/App.tsx b/src/App.tsx new file mode 100644 index 0000000..d1f0cfb --- /dev/null +++ b/src/App.tsx @@ -0,0 +1,34 @@ +import { BrowserRouter, Routes, Route, Navigate } from "react-router-dom"; +import AppLayout from "./components/layout/AppLayout"; +import ProjectForm from "./components/projects/ProjectForm"; +import ProjectDashboard from "./components/projects/ProjectDashboard"; +import SettingsPage from "./components/settings/SettingsPage"; +import TrackerConfig from "./components/trackers/TrackerConfig"; + +function EmptyState() { + return ( +
+

Select a project or create a new one

+
+ ); +} + +function App() { + return ( + + + }> + } /> + } /> + } /> + } /> + } /> + } /> + } /> + + + + ); +} + +export default App; diff --git a/src/components/layout/AppLayout.tsx b/src/components/layout/AppLayout.tsx new file mode 100644 index 0000000..7703abd --- /dev/null +++ b/src/components/layout/AppLayout.tsx @@ -0,0 +1,13 @@ +import { Outlet } from "react-router-dom"; +import Sidebar from "./Sidebar"; + +export default function AppLayout() { + return ( +
+ +
+ +
+
+ ); +} diff --git a/src/components/layout/Sidebar.tsx b/src/components/layout/Sidebar.tsx new file mode 100644 index 0000000..7d1d943 --- /dev/null +++ b/src/components/layout/Sidebar.tsx @@ -0,0 +1,69 @@ +import { useEffect, useState } from "react"; +import { Link, useParams } from "react-router-dom"; +import { listProjects } from "../../lib/api"; +import type { Project } from "../../lib/types"; + +export default function Sidebar() { + const [projects, setProjects] = useState([]); + const { projectId } = useParams(); + + useEffect(() => { + listProjects().then(setProjects); + }, []); + + // Expose a refresh function via custom event + useEffect(() => { + const handler = () => { + listProjects().then(setProjects); + }; + window.addEventListener("orchai:refresh-projects", handler); + return () => window.removeEventListener("orchai:refresh-projects", handler); + }, []); + + return ( + + ); +} diff --git a/src/components/projects/ProjectDashboard.tsx b/src/components/projects/ProjectDashboard.tsx new file mode 100644 index 0000000..8e0b089 --- /dev/null +++ b/src/components/projects/ProjectDashboard.tsx @@ -0,0 +1,133 @@ +import { useEffect, useState } from "react"; +import { useParams, Link, useNavigate } from "react-router-dom"; +import { getProject, deleteProject, listTrackers, listProcessedTickets } from "../../lib/api"; +import type { Project, WatchedTracker, ProcessedTicket } from "../../lib/types"; +import TrackerList from "../trackers/TrackerList"; + +export default function ProjectDashboard() { + const { projectId } = useParams(); + const navigate = useNavigate(); + const [project, setProject] = useState(null); + const [trackers, setTrackers] = useState([]); + const [tickets, setTickets] = useState([]); + + async function loadData() { + if (!projectId) return; + const [proj, trks, tkts] = await Promise.all([ + getProject(projectId), + listTrackers(projectId), + listProcessedTickets(projectId), + ]); + setProject(proj); + setTrackers(trks); + setTickets(tkts); + } + + useEffect(() => { + loadData(); + }, [projectId]); + + async function handleDelete() { + if (!projectId) return; + if (!window.confirm(`Delete project "${project?.name}"?`)) return; + + await deleteProject(projectId); + window.dispatchEvent(new Event("orchai:refresh-projects")); + navigate("/"); + } + + function statusBadgeClass(status: string): string { + switch (status) { + case "Pending": + return "bg-yellow-100 text-yellow-700"; + case "Done": + return "bg-green-100 text-green-700"; + case "Error": + return "bg-red-100 text-red-700"; + default: + return "bg-blue-100 text-blue-700"; + } + } + + if (!project) { + return
Loading...
; + } + + const recentTickets = tickets.slice(-10).reverse(); + + return ( +
+
+

{project.name}

+
+ + Edit + + +
+
+ +
+
+ Path: + {project.path} +
+ {project.cloned_from && ( +
+ Cloned from: + {project.cloned_from} +
+ )} +
+ Base branch: + {project.base_branch} +
+
+ Created: + {new Date(project.created_at).toLocaleDateString()} +
+
+ +
+

Watched Trackers

+ +
+ +
+

Recent Tickets

+ {recentTickets.length === 0 ? ( +
No tickets processed yet.
+ ) : ( +
+ {recentTickets.map((ticket) => ( +
+
+
+ #{ticket.artifact_id} + {ticket.artifact_title} +
+
+ + {ticket.status} + +
+ ))} +
+ )} +
+
+ ); +} diff --git a/src/components/projects/ProjectForm.tsx b/src/components/projects/ProjectForm.tsx new file mode 100644 index 0000000..6c46845 --- /dev/null +++ b/src/components/projects/ProjectForm.tsx @@ -0,0 +1,174 @@ +import { useState, useEffect } from "react"; +import { useNavigate, useParams } from "react-router-dom"; +import { open } from "@tauri-apps/plugin-dialog"; +import { createProject, getProject, updateProject } from "../../lib/api"; + +export default function ProjectForm() { + const navigate = useNavigate(); + const { projectId } = useParams(); + const isEditing = Boolean(projectId); + + const [name, setName] = useState(""); + const [pathOrUrl, setPathOrUrl] = useState(""); + const [baseBranch, setBaseBranch] = useState("main"); + const [mode, setMode] = useState<"local" | "clone">("local"); + const [error, setError] = useState(null); + const [loading, setLoading] = useState(false); + + useEffect(() => { + if (projectId) { + getProject(projectId).then((project) => { + setName(project.name); + setPathOrUrl(project.path); + setBaseBranch(project.base_branch); + if (project.cloned_from) { + setMode("clone"); + } + }); + } + }, [projectId]); + + async function handleBrowse() { + const selected = await open({ directory: true, multiple: false }); + if (selected) { + setPathOrUrl(selected as string); + } + } + + async function handleSubmit(e: React.FormEvent) { + e.preventDefault(); + setError(null); + setLoading(true); + + try { + if (isEditing && projectId) { + await updateProject(projectId, name, baseBranch); + } else { + await createProject(name, pathOrUrl, baseBranch); + } + window.dispatchEvent(new Event("orchai:refresh-projects")); + navigate("/"); + } catch (err: unknown) { + const message = err instanceof Error ? err.message : String(err); + setError(message); + } finally { + setLoading(false); + } + } + + return ( +
+

+ {isEditing ? "Edit project" : "New project"} +

+ +
+
+ + setName(e.target.value)} + required + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+ + {!isEditing && ( + <> +
+ +
+ + +
+
+ +
+ +
+ setPathOrUrl(e.target.value)} + required + placeholder={ + mode === "local" + ? "/home/user/code/myproject" + : "https://github.com/org/repo.git" + } + className="flex-1 border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> + {mode === "local" && ( + + )} +
+
+ + )} + +
+ + setBaseBranch(e.target.value)} + required + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+ + {error && ( +
+ {error} +
+ )} + +
+ + +
+
+
+ ); +} diff --git a/src/components/settings/SettingsPage.tsx b/src/components/settings/SettingsPage.tsx new file mode 100644 index 0000000..170a092 --- /dev/null +++ b/src/components/settings/SettingsPage.tsx @@ -0,0 +1,174 @@ +import { useState, useEffect } from "react"; +import { + getTuleapCredentials, + setTuleapCredentials, + deleteTuleapCredentials, + testTuleapConnection, +} from "../../lib/api"; +import type { TuleapCredentialsSafe } from "../../lib/types"; + +export default function SettingsPage() { + const [tuleapUrl, setTuleapUrl] = useState(""); + const [username, setUsername] = useState(""); + const [password, setPassword] = useState(""); + const [existing, setExisting] = useState(null); + const [error, setError] = useState(null); + const [success, setSuccess] = useState(null); + const [saving, setSaving] = useState(false); + const [testing, setTesting] = useState(false); + const [deleting, setDeleting] = useState(false); + + useEffect(() => { + getTuleapCredentials().then((creds) => { + if (creds) { + setExisting(creds); + setTuleapUrl(creds.tuleap_url); + setUsername(creds.username); + } + }); + }, []); + + function clearMessages() { + setError(null); + setSuccess(null); + } + + async function handleSave(e: React.FormEvent) { + e.preventDefault(); + clearMessages(); + setSaving(true); + try { + const creds = await setTuleapCredentials(tuleapUrl, username, password); + setExisting(creds); + setPassword(""); + setSuccess("Credentials saved."); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setSaving(false); + } + } + + async function handleTest() { + clearMessages(); + setTesting(true); + try { + const msg = await testTuleapConnection(); + setSuccess(msg); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setTesting(false); + } + } + + async function handleDelete() { + if (!window.confirm("Delete Tuleap credentials? This cannot be undone.")) return; + clearMessages(); + setDeleting(true); + try { + await deleteTuleapCredentials(); + setExisting(null); + setTuleapUrl(""); + setUsername(""); + setPassword(""); + setSuccess("Credentials deleted."); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setDeleting(false); + } + } + + return ( +
+

Settings

+ +
+

Tuleap credentials

+ +
+
+ + setTuleapUrl(e.target.value)} + required + placeholder="https://tuleap.example.com" + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+ +
+ + setUsername(e.target.value)} + required + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+ +
+ + setPassword(e.target.value)} + placeholder={existing ? "Leave empty to keep current" : ""} + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+ + {error && ( +
+ {error} +
+ )} + + {success && ( +
+ {success} +
+ )} + +
+ + + {existing && ( + + )} +
+
+
+
+ ); +} diff --git a/src/components/trackers/FilterBuilder.tsx b/src/components/trackers/FilterBuilder.tsx new file mode 100644 index 0000000..45ad64f --- /dev/null +++ b/src/components/trackers/FilterBuilder.tsx @@ -0,0 +1,182 @@ +import type { FilterGroup, Filter, TrackerField } from "../../lib/types"; + +const OPERATORS = ["In", "NotIn", "Equals", "NotEquals"] as const; + +interface Props { + groups: FilterGroup[]; + onChange: (groups: FilterGroup[]) => void; + availableFields: TrackerField[]; +} + +export default function FilterBuilder({ groups, onChange, availableFields }: Props) { + function updateGroup(groupIndex: number, group: FilterGroup) { + const next = groups.map((g, i) => (i === groupIndex ? group : g)); + onChange(next); + } + + function removeGroup(groupIndex: number) { + onChange(groups.filter((_, i) => i !== groupIndex)); + } + + function addGroup() { + onChange([...groups, { conditions: [{ field: "", operator: "In", value: [] }] }]); + } + + function updateCondition(groupIndex: number, condIndex: number, cond: Filter) { + const group = groups[groupIndex]; + const conditions = group.conditions.map((c, i) => (i === condIndex ? cond : c)); + updateGroup(groupIndex, { conditions }); + } + + function removeCondition(groupIndex: number, condIndex: number) { + const group = groups[groupIndex]; + const conditions = group.conditions.filter((_, i) => i !== condIndex); + updateGroup(groupIndex, { conditions }); + } + + function addCondition(groupIndex: number) { + const group = groups[groupIndex]; + updateGroup(groupIndex, { + conditions: [...group.conditions, { field: "", operator: "In", value: [] }], + }); + } + + function toggleValue(groupIndex: number, condIndex: number, val: string) { + const cond = groups[groupIndex].conditions[condIndex]; + const next = cond.value.includes(val) + ? cond.value.filter((v) => v !== val) + : [...cond.value, val]; + updateCondition(groupIndex, condIndex, { ...cond, value: next }); + } + + return ( +
+ {groups.map((group, gi) => ( +
+ {gi > 0 && ( +
+
+ AND +
+
+ )} +
+
+ Group {gi + 1} + +
+ +
+ {group.conditions.map((cond, ci) => { + const fieldDef = availableFields.find((f) => f.label === cond.field); + return ( +
+ {ci > 0 && ( +
+
+ OR +
+
+ )} +
+
+ {/* Field dropdown */} + + + {/* Operator dropdown */} + + + {/* Remove condition */} + +
+ + {/* Value pills */} + {fieldDef && fieldDef.values.length > 0 && ( +
+ {fieldDef.values.map((v) => { + const selected = cond.value.includes(String(v.id)); + return ( + + ); + })} +
+ )} +
+
+ ); + })} +
+ + +
+
+ ))} + + +
+ ); +} diff --git a/src/components/trackers/TrackerConfig.tsx b/src/components/trackers/TrackerConfig.tsx new file mode 100644 index 0000000..8b05c15 --- /dev/null +++ b/src/components/trackers/TrackerConfig.tsx @@ -0,0 +1,190 @@ +import { useState } from "react"; +import { useParams, useNavigate } from "react-router-dom"; +import { addTracker, getTrackerFields } from "../../lib/api"; +import type { FilterGroup, TrackerField, AgentConfig } from "../../lib/types"; +import FilterBuilder from "./FilterBuilder"; + +export default function TrackerConfig() { + const { projectId } = useParams<{ projectId: string }>(); + const navigate = useNavigate(); + + const [trackerId, setTrackerId] = useState(""); + const [trackerLabel, setTrackerLabel] = useState(""); + const [pollingInterval, setPollingInterval] = useState(10); + const [fields, setFields] = useState([]); + const [fieldsLoaded, setFieldsLoaded] = useState(false); + const [fieldsLoading, setFieldsLoading] = useState(false); + const [filters, setFilters] = useState([]); + const [analystCommand, setAnalystCommand] = useState("claude"); + const [developerCommand, setDeveloperCommand] = useState("claude"); + const [error, setError] = useState(null); + const [loading, setLoading] = useState(false); + + async function handleLoadFields() { + if (!trackerId) return; + setFieldsLoading(true); + setError(null); + try { + const result = await getTrackerFields(Number(trackerId)); + setFields(result); + setFieldsLoaded(true); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setFieldsLoading(false); + } + } + + async function handleSubmit(e: React.FormEvent) { + e.preventDefault(); + if (!projectId || trackerId === "") return; + setError(null); + setLoading(true); + + const agentConfig: AgentConfig = { + analyst_command: analystCommand, + analyst_args: [], + developer_command: developerCommand, + developer_args: [], + }; + + try { + await addTracker(projectId, Number(trackerId), trackerLabel, pollingInterval, agentConfig, filters); + navigate(`/projects/${projectId}`); + } catch (err: unknown) { + setError(err instanceof Error ? err.message : String(err)); + } finally { + setLoading(false); + } + } + + return ( +
+

Add tracker

+ +
+ {/* Basic fields */} +
+
+ +
+ { + setTrackerId(e.target.value === "" ? "" : Number(e.target.value)); + setFieldsLoaded(false); + setFields([]); + }} + required + min={1} + className="w-40 border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + placeholder="e.g. 42" + /> + +
+
+ +
+ + setTrackerLabel(e.target.value)} + required + placeholder="e.g. Bugs" + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+ +
+ + setPollingInterval(Number(e.target.value))} + required + min={1} + className="w-40 border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+
+ + {/* Filter builder */} + {fieldsLoaded && ( +
+

Filters

+ +
+ )} + + {/* Agent config */} +
+

Agent configuration

+
+ + setAnalystCommand(e.target.value)} + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+
+ + setDeveloperCommand(e.target.value)} + className="w-full border border-gray-300 rounded px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500" + /> +
+
+ + {error && ( +
+ {error} +
+ )} + +
+ + +
+
+
+ ); +} diff --git a/src/components/trackers/TrackerList.tsx b/src/components/trackers/TrackerList.tsx new file mode 100644 index 0000000..fd03506 --- /dev/null +++ b/src/components/trackers/TrackerList.tsx @@ -0,0 +1,112 @@ +import { Link } from "react-router-dom"; +import { manualPoll, updateTracker, removeTracker } from "../../lib/api"; +import type { WatchedTracker } from "../../lib/types"; + +interface Props { + trackers: WatchedTracker[]; + projectId: string; + onRefresh: () => void; +} + +export default function TrackerList({ trackers, projectId, onRefresh }: Props) { + async function handlePollNow(tracker: WatchedTracker) { + try { + await manualPoll(tracker.id); + onRefresh(); + } catch (err) { + console.error("Poll failed:", err); + } + } + + async function handleToggleEnabled(tracker: WatchedTracker) { + try { + await updateTracker( + tracker.id, + tracker.polling_interval, + tracker.agent_config, + tracker.filters, + !tracker.enabled + ); + onRefresh(); + } catch (err) { + console.error("Update failed:", err); + } + } + + async function handleRemove(tracker: WatchedTracker) { + if (!window.confirm(`Remove tracker "${tracker.tracker_label}"?`)) return; + try { + await removeTracker(tracker.id); + onRefresh(); + } catch (err) { + console.error("Remove failed:", err); + } + } + + return ( +
+ {trackers.length === 0 && ( +
No trackers configured.
+ )} + + {trackers.map((tracker) => ( +
+
+
+ {tracker.tracker_label} + #{tracker.tracker_id} + + {tracker.enabled ? "Active" : "Paused"} + +
+
+ {tracker.last_polled_at + ? `Last poll: ${new Date(tracker.last_polled_at).toLocaleString()}` + : "Never polled"} +
+
+ +
+ + + +
+
+ ))} + + + Add tracker + +
+ ); +} diff --git a/src/index.css b/src/index.css new file mode 100644 index 0000000..f1d8c73 --- /dev/null +++ b/src/index.css @@ -0,0 +1 @@ +@import "tailwindcss"; diff --git a/src/lib/api.ts b/src/lib/api.ts new file mode 100644 index 0000000..ca6e5e0 --- /dev/null +++ b/src/lib/api.ts @@ -0,0 +1,84 @@ +import { invoke } from "@tauri-apps/api/core"; +import type { + Project, + TuleapCredentialsSafe, + AgentConfig, + FilterGroup, + WatchedTracker, + TrackerField, + ProcessedTicket, +} from "./types"; + +export async function createProject( + name: string, + pathOrUrl: string, + baseBranch: string +): Promise { + return invoke("create_project", { + name, + pathOrUrl, + baseBranch, + }); +} + +export async function listProjects(): Promise { + return invoke("list_projects"); +} + +export async function getProject(id: string): Promise { + return invoke("get_project", { id }); +} + +export async function updateProject( + id: string, + name: string, + baseBranch: string +): Promise { + return invoke("update_project", { id, name, baseBranch }); +} + +export async function deleteProject(id: string): Promise { + return invoke("delete_project", { id }); +} + +// Credentials +export async function setTuleapCredentials(tuleapUrl: string, username: string, password: string): Promise { + return invoke("set_tuleap_credentials", { tuleapUrl, username, password }); +} +export async function getTuleapCredentials(): Promise { + return invoke("get_tuleap_credentials"); +} +export async function deleteTuleapCredentials(): Promise { + return invoke("delete_tuleap_credentials"); +} +export async function testTuleapConnection(): Promise { + return invoke("test_tuleap_connection"); +} + +// Trackers +export async function addTracker(projectId: string, trackerId: number, trackerLabel: string, pollingInterval: number, agentConfig: AgentConfig, filters: FilterGroup[]): Promise { + return invoke("add_tracker", { projectId, trackerId, trackerLabel, pollingInterval, agentConfig, filters }); +} +export async function listTrackers(projectId: string): Promise { + return invoke("list_trackers", { projectId }); +} +export async function updateTracker(id: string, pollingInterval: number, agentConfig: AgentConfig, filters: FilterGroup[], enabled: boolean): Promise { + return invoke("update_tracker", { id, pollingInterval, agentConfig, filters, enabled }); +} +export async function removeTracker(id: string): Promise { + return invoke("remove_tracker", { id }); +} +export async function getTrackerFields(trackerId: number): Promise { + return invoke("get_tracker_fields", { trackerId }); +} + +// Tickets & Polling +export async function listProcessedTickets(projectId: string): Promise { + return invoke("list_processed_tickets", { projectId }); +} +export async function manualPoll(trackerId: string): Promise { + return invoke("manual_poll", { trackerId }); +} +export async function getQueueStatus(projectId: string): Promise { + return invoke("get_queue_status", { projectId }); +} diff --git a/src/lib/types.ts b/src/lib/types.ts new file mode 100644 index 0000000..85ebc7d --- /dev/null +++ b/src/lib/types.ts @@ -0,0 +1,71 @@ +export interface Project { + id: string; + name: string; + path: string; + cloned_from: string | null; + base_branch: string; + created_at: string; +} + +export interface TuleapCredentialsSafe { + id: string; + tuleap_url: string; + username: string; +} + +export interface AgentConfig { + analyst_command: string; + analyst_args: string[]; + developer_command: string; + developer_args: string[]; +} + +export interface Filter { + field: string; + operator: string; + value: string[]; +} + +export interface FilterGroup { + conditions: Filter[]; +} + +export interface TrackerField { + field_id: number; + label: string; + field_type: string; + values: FieldValue[]; +} + +export interface FieldValue { + id: number; + label: string; +} + +export interface WatchedTracker { + id: string; + project_id: string; + tracker_id: number; + tracker_label: string; + polling_interval: number; + agent_config: AgentConfig; + filters: FilterGroup[]; + enabled: boolean; + last_polled_at: string | null; + created_at: string; +} + +export interface ProcessedTicket { + id: string; + tracker_id: string; + artifact_id: number; + artifact_title: string; + artifact_data: string; + status: string; + analyst_report: string | null; + developer_report: string | null; + worktree_path: string | null; + branch_name: string | null; + detected_at: string; + processed_at: string | null; +} diff --git a/src/main.tsx b/src/main.tsx new file mode 100644 index 0000000..02055fd --- /dev/null +++ b/src/main.tsx @@ -0,0 +1,10 @@ +import React from "react"; +import ReactDOM from "react-dom/client"; +import App from "./App"; +import "./index.css"; + +ReactDOM.createRoot(document.getElementById("root") as HTMLElement).render( + + + +); diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..f0a2350 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,24 @@ +{ + "compilerOptions": { + "target": "ES2020", + "useDefineForClassFields": true, + "lib": ["ES2020", "DOM", "DOM.Iterable"], + "module": "ESNext", + "skipLibCheck": true, + + /* Bundler mode */ + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "isolatedModules": true, + "moduleDetection": "force", + "noEmit": true, + "jsx": "react-jsx", + + /* Linting */ + "strict": true, + "noUnusedLocals": true, + "noUnusedParameters": true, + "noFallthroughCasesInSwitch": true + }, + "include": ["src"] +} diff --git a/vite.config.ts b/vite.config.ts new file mode 100644 index 0000000..c89f997 --- /dev/null +++ b/vite.config.ts @@ -0,0 +1,25 @@ +import { defineConfig } from "vite"; +import react from "@vitejs/plugin-react"; +import tailwindcss from "@tailwindcss/vite"; + +const host = process.env.TAURI_DEV_HOST; + +export default defineConfig(async () => ({ + plugins: [react(), tailwindcss()], + clearScreen: false, + server: { + port: 1420, + strictPort: true, + host: host || false, + hmr: host + ? { + protocol: "ws", + host, + port: 1421, + } + : undefined, + watch: { + ignored: ["**/src-tauri/**"], + }, + }, +}));