diff --git a/docs/reference-repos/README.md b/docs/reference-repos/README.md new file mode 100644 index 0000000..33381e1 --- /dev/null +++ b/docs/reference-repos/README.md @@ -0,0 +1,69 @@ +# Reference Repos + +Third-party Dune: Awakening server-management projects, kept here as **behavior +references** for Phase 2 (the Corrosion host-agent Dune adapter + future panel +Dune features). These are NOT Corrosion code and are not built or shipped — they +are read-only references. `.git` histories, `node_modules`, and compiled +binaries were stripped on import (the 38 MB `icehunter/web/dune-admin` build +artifact and a Tauri `.icns` are intentionally absent). + +> Imported 2026-06-12 from `/tmp/dune-re`. Each was a separate upstream repo; +> see each project's own `LICENSE` and `README.md`. Treat as documentation. + +## Why these are here + +Dune: Awakening does **not** use SteamCMD or a plain game-server process like +Rust/Conan/Soulmask. It ships as **Docker container(s)** fronted by a **RabbitMQ +broker** (admin + game vhosts) and a **PostgreSQL** admin database (`dune` +schema), orchestrated as a "**battlegroup**". The game process is +`DuneSandboxServer-Linux-Shipping` (one per partition). Server settings live in +INI files (`UserEngine.ini` / `UserGame.ini`) and only take effect after a +restart. Our Dune adapter must model that container/broker/DB world instead of +the process+SteamCMD model — these repos are how that world actually works in +the wild. + +## The references + +### `icehunter/` — `dune-admin` (Go backend + React SPA) +The richest ops reference. A web admin panel with **four interchangeable control +planes**: `docker`, `kubectl`, `local`, and `amp` (CubeCoders AMP / podman). +Most relevant to us: +- **`SETUP_DOCKER.md`** — the Docker control plane: `docker start/stop/restart` + for lifecycle, `docker logs -f` for streaming, `docker exec` into the broker + container for RabbitMQ (`rabbitmqctl`) commands, direct TCP to the `dune` + Postgres. Optional SSH tunnelling when the admin is off-host. **This is the + closest analog to what the Corrosion host-agent Dune adapter must do.** +- `cmd/dune-admin/control_docker.go` / `control_kubectl.go` / `control_local.go` + / `control_amp.go` — the `ControlPlane` interface and its implementations + (the start/stop/restart/status/log/broker abstraction we mirror as a Rust + game-adapter trait). +- `db.go` / `model.go` — the full Dune admin data model (players, bases, + inventory, exchange/market) for when Corrosion grows a richer Dune admin + surface beyond lifecycle. +- `CLAUDE.md` — upstream's own engineering notes; the AMP section documents the + INI-vs-API server-settings gotcha (AMP regenerates INIs on start). + +### `adainrivers/` — Dune Dedicated Server Manager (Rust / Tauri desktop) +**The Rust reference.** Manages already-provisioned servers over **SSH + +Kubernetes** ("BattleGroup" start/stop/restart/update), with secure SSH tunnels +to Director / File Browser / Postgres / PgHero, an in-game admin console (item +grants, vehicle spawns, journey/XP tags), and a bundled **`dune-server-service`** +daemon for scheduled maintenance (timed restarts with in-game warnings, backups, +update apply). Closest to our stack idiomatically — read it for Rust patterns on +SSH control, the maintenance-daemon design, and the in-game command surface. + +### `the4rchangel/` — Dune: Awakening Server Manager (Node.js local web UI) +**Matches the Commander's exact self-host path.** A local dashboard that +replaces the `battlegroup.bat` terminal menu — guided VM import (Hyper-V), +network, SSH, bootstrap, then daily ops: battlegroup start/stop/restart/update, +character editor, visual game-config editor (PvP, sandstorms, sandworms, mining +rates, decay, building limits), monitoring, DB access. Read it to understand the +`battlegroup.bat` workflow our agent has to drive on a Windows/Hyper-V host. + +## How we use them + +- **Lifecycle/control** → mirror `icehunter`'s `ControlPlane` docker provider as + the agent's Dune game-adapter (compose/`docker` lifecycle, `docker logs` + console, reject SteamCMD). +- **Rust idioms / maintenance daemon / SSH** → `adainrivers`. +- **Battlegroup.bat reality / setup flow / game-config schema** → `the4rchangel`. diff --git a/docs/reference-repos/adainrivers/.github/workflows/ci.yml b/docs/reference-repos/adainrivers/.github/workflows/ci.yml new file mode 100644 index 0000000..60e6376 --- /dev/null +++ b/docs/reference-repos/adainrivers/.github/workflows/ci.yml @@ -0,0 +1,71 @@ +name: CI + +on: + workflow_dispatch: + +env: + FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: true + +jobs: + checks: + name: Workspace checks (${{ matrix.platform }}) + runs-on: ${{ matrix.platform }} + strategy: + fail-fast: false + matrix: + platform: [windows-latest, ubuntu-22.04, macos-latest] + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Install Rust + uses: dtolnay/rust-toolchain@stable + + - name: Install Node + uses: actions/setup-node@v4 + with: + node-version: 22 + cache: npm + cache-dependency-path: app/package-lock.json + + - name: Install Linux Tauri dependencies + if: matrix.platform == 'ubuntu-22.04' + run: | + sudo apt-get update + sudo apt-get install -y libwebkit2gtk-4.1-dev libappindicator3-dev librsvg2-dev patchelf pkg-config libssl-dev + + - name: Install frontend dependencies + working-directory: app + run: npm ci + + - name: Rust format + run: cargo fmt --all -- --check + + - name: Rust check + run: cargo check --workspace + + - name: Rust tests + run: cargo test --workspace + + - name: Core API docs + run: cargo doc -p dune-manager-core --no-deps + + - name: Frontend build + working-directory: app + run: npm run build + + - name: Tauri shell check + run: cargo check -p dune-dedicated-server-manager-app + + - name: Secret and machine-constant scan + if: matrix.platform == 'windows-latest' + shell: pwsh + run: | + rg -n -S "I:|AutoUpdate|192\.168\.2\.|menna|dune-awakening|C:\\WINDOWS\\System32\\OpenSSH|C:\\Windows\\System32\\OpenSSH|change-me-before-exposing|c05564d|d177d3bbc40be761|qRmQx|FuncomLiveServices__ServiceAuthToken" . -g "!app/**/target/**" -g "!crates/**/target/**" -g "!target/**" -g "!app/node_modules/**" -g "!app/dist/**" -g "!*.md" -g "!app/steamcmd/**" -g "!app/dune-server/**" -g "!app/vm/**" -g "!app/vm-*/**" -g "!vm/**" -g "!.tmp/**" + if ($LASTEXITCODE -eq 0) { + throw "Secret or machine-specific constant scan found matches." + } + if ($LASTEXITCODE -ne 1) { + exit $LASTEXITCODE + } diff --git a/docs/reference-repos/adainrivers/.github/workflows/release.yml b/docs/reference-repos/adainrivers/.github/workflows/release.yml new file mode 100644 index 0000000..d0af4db --- /dev/null +++ b/docs/reference-repos/adainrivers/.github/workflows/release.yml @@ -0,0 +1,203 @@ +name: Release + +on: + push: + tags: + - "v*.*.*" + workflow_dispatch: + inputs: + version: + description: "Version to release, for example 0.1.0" + required: true + type: string + +permissions: + contents: write + +env: + FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: true + +jobs: + linux-service-binary: + name: Build dune-server-service (musl) + runs-on: ubuntu-22.04 + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Install Rust toolchain + uses: dtolnay/rust-toolchain@stable + with: + targets: x86_64-unknown-linux-musl + + - name: Install Zig + uses: mlugg/setup-zig@v1 + with: + version: 0.13.0 + + - name: Install cargo-zigbuild + run: cargo install --locked cargo-zigbuild + + - name: Resolve release version + shell: bash + env: + WORKFLOW_VERSION: ${{ inputs.version }} + run: | + version="$WORKFLOW_VERSION" + if [ -z "$version" ]; then + version="${GITHUB_REF_NAME#v}" + fi + if [ -z "$version" ]; then + echo "could not resolve release version" >&2 + exit 1 + fi + echo "RELEASE_VERSION=$version" >> "$GITHUB_ENV" + echo "RELEASE_TAG=v$version" >> "$GITHUB_ENV" + + - name: Build musl binary + run: | + cargo zigbuild -p dune-server-service --release --target x86_64-unknown-linux-musl + strip target/x86_64-unknown-linux-musl/release/dune-server-service + + - name: Stage release artifacts + run: | + mkdir -p release-artifacts + cp target/x86_64-unknown-linux-musl/release/dune-server-service release-artifacts/dune-server-service + cp crates/dune-server-service/systemd/dune-server-service.service release-artifacts/dune-server-service.service + cp crates/dune-server-service/openrc/dune-server-service release-artifacts/dune-server-service.openrc + + - name: Upload artifact for desktop bundle + uses: actions/upload-artifact@v4 + with: + name: dune-server-service-musl + path: release-artifacts/ + retention-days: 7 + + - name: Resolve release notes + if: startsWith(github.ref, 'refs/tags/v') + shell: bash + run: | + notes_path="release-notes/${RELEASE_VERSION}.md" + if [ -f "$notes_path" ]; then + echo "RELEASE_BODY_PATH=$notes_path" >> "$GITHUB_ENV" + else + tmp=$(mktemp) + printf 'Release v%s. No release-notes/%s.md was provided — see the commit log for details.\n' \ + "$RELEASE_VERSION" "$RELEASE_VERSION" > "$tmp" + echo "RELEASE_BODY_PATH=$tmp" >> "$GITHUB_ENV" + fi + + - name: Attach to GitHub release + if: startsWith(github.ref, 'refs/tags/v') + uses: softprops/action-gh-release@v2 + with: + tag_name: ${{ env.RELEASE_TAG }} + body_path: ${{ env.RELEASE_BODY_PATH }} + files: | + release-artifacts/dune-server-service + release-artifacts/dune-server-service.service + release-artifacts/dune-server-service.openrc + + desktop-app: + name: Build ${{ matrix.name }} app + needs: linux-service-binary + runs-on: ${{ matrix.platform }} + strategy: + fail-fast: false + matrix: + include: + - name: Windows + platform: windows-latest + args: --bundles nsis + - name: Linux + platform: ubuntu-22.04 + args: --bundles appimage,deb + - name: macOS Apple Silicon + platform: macos-latest + args: --target aarch64-apple-darwin --bundles dmg + - name: macOS Intel + platform: macos-latest + args: --target x86_64-apple-darwin --bundles dmg + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Install Rust + uses: dtolnay/rust-toolchain@stable + with: + targets: ${{ startsWith(matrix.name, 'macOS') && 'aarch64-apple-darwin,x86_64-apple-darwin' || '' }} + + - name: Install Node + uses: actions/setup-node@v4 + with: + node-version: 22 + cache: npm + cache-dependency-path: app/package-lock.json + + - name: Install Linux Tauri dependencies + if: matrix.platform == 'ubuntu-22.04' + run: | + sudo apt-get update + sudo apt-get install -y libwebkit2gtk-4.1-dev libappindicator3-dev librsvg2-dev patchelf pkg-config libssl-dev + + - name: Install frontend dependencies + working-directory: app + run: npm ci + + - name: Download bundled dune-server-service binary + uses: actions/download-artifact@v4 + with: + name: dune-server-service-musl + path: app/src-tauri/binaries/ + + - name: Resolve release version + shell: pwsh + env: + WORKFLOW_VERSION: ${{ inputs.version }} + run: | + $version = $env:WORKFLOW_VERSION + if ([string]::IsNullOrWhiteSpace($version)) { + $version = "${{ github.ref_name }}".TrimStart("v") + } + if ([string]::IsNullOrWhiteSpace($version)) { + throw "Release version could not be resolved." + } + "RELEASE_VERSION=$version" | Out-File -FilePath $env:GITHUB_ENV -Append + "RELEASE_TAG=v$version" | Out-File -FilePath $env:GITHUB_ENV -Append + + - name: Prepare release config + shell: pwsh + run: | + $version = $env:RELEASE_VERSION + + Push-Location app + npm version --no-git-tag-version --allow-same-version $version + Pop-Location + + $tauriConfigPath = "app/src-tauri/tauri.conf.json" + $config = Get-Content $tauriConfigPath -Raw + $config = $config -replace '"version":\s*"[^"]+"', ('"version": "' + $version + '"') + # Release builds publish signed updater artifacts; the checked-in + # default keeps this off so local debug builds do not require + # TAURI_SIGNING_PRIVATE_KEY. + $config = $config -replace '"createUpdaterArtifacts":\s*false', '"createUpdaterArtifacts": true' + Set-Content -Path $tauriConfigPath -Value $config -NoNewline + + # The body is set by the linux-service-binary job's softprops step. + # tauri-action only uploads desktop bundles + the signed updater + # artifacts here; we don't pass releaseBody to avoid clobbering. + - name: Build and publish Tauri release + uses: tauri-apps/tauri-action@v0 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY }} + TAURI_SIGNING_PRIVATE_KEY_PASSWORD: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY_PASSWORD }} + VITE_ENABLE_STARTUP_UPDATE_CHECK: "true" + with: + projectPath: app + tagName: ${{ env.RELEASE_TAG }} + releaseName: "Dune Dedicated Server Manager ${{ env.RELEASE_TAG }}" + releaseDraft: false + prerelease: false + args: ${{ matrix.args }} diff --git a/docs/reference-repos/adainrivers/.gitignore b/docs/reference-repos/adainrivers/.gitignore new file mode 100644 index 0000000..b8d6659 --- /dev/null +++ b/docs/reference-repos/adainrivers/.gitignore @@ -0,0 +1,68 @@ +# Dependencies +node_modules/ +app/node_modules/ + +# Frontend build +dist/ +app/dist/ +app/src-tauri/gen/schemas/ + +# Rust/Tauri build outputs +target/ +src-tauri/target/ +app/src-tauri/target/ +manager-api/target/ + +# Local environment +.env +.env.* +!.env.example + +# Logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +pnpm-debug.log* + +# Docs are scratch notes for now; keep README trackable later +*.md +!README.md +!docs/ +!docs/*.md +docs/rabbitmq-protocol.md +# Release notes go on GitHub releases via the release workflow. +!release-notes/ +!release-notes/*.md + +# Editor and OS noise +.idea/ +.vscode/ +*.swp +*.swo +Thumbs.db +Desktop.ini + +# Local app/runtime data and secrets +.tmp/ +.playwright-mcp/ +app/default-config.json +app/steamcmd/ +app/dune-server/ +dune-server/ +app/vm/ +app/vm-*/ +app/src-tauri/dune-server/ +app/src-tauri/vm/ +app/src-tauri/resources/manager-api/dune-manager-api +app/src-tauri/resources/manager-api/dune-manager-api.exe +vm/ +*.pem +*.key +sshKey +codex_vm_ed25519_dropbear +codex_vm_ed25519_dropbear.pub +snapshots/ +keys/ +initial-setup-log.txt +secrets/ diff --git a/docs/reference-repos/adainrivers/Cargo.lock b/docs/reference-repos/adainrivers/Cargo.lock new file mode 100644 index 0000000..ada95f5 --- /dev/null +++ b/docs/reference-repos/adainrivers/Cargo.lock @@ -0,0 +1,7156 @@ +# 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.6.0-rc.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6b657e772794c6b04730ea897b66a058ccd866c16d1967da05eeeecec39043fe" +dependencies = [ + "crypto-common 0.2.2", + "inout 0.2.2", +] + +[[package]] +name = "aes" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "66bd29a732b644c0431c6140f370d097879203d79b80c94a6747ba0872adaef8" +dependencies = [ + "cipher", + "cpubits", + "cpufeatures 0.3.0", + "zeroize", +] + +[[package]] +name = "aes-gcm" +version = "0.11.0-rc.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e22c0c90bbe8d4f77c3ca9ddabe41a1f8382d6fc1f7cea89459d0f320371f972" +dependencies = [ + "aead", + "aes", + "cipher", + "ctr", + "ghash", + "subtle", + "zeroize", +] + +[[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 = "arbitrary" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3d036a3c4ab069c7b410a2ce876bd74808d2d0888a82667669f8e783a898bf1" +dependencies = [ + "derive_arbitrary", +] + +[[package]] +name = "argon2" +version = "0.6.0-rc.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7af50940b73bf4e16c15c448a2b121c63f2d68e3e54b6a8731673cb4aa0cdff5" +dependencies = [ + "base64ct", + "blake2", + "cpufeatures 0.3.0", + "password-hash", +] + +[[package]] +name = "async-trait" +version = "0.1.89" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9035ad2d096bed7955a320ee7e2230574d28fd3c3a0f186cbea1ff3c7eed5dbb" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[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 = "axum" +version = "0.7.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "edca88bc138befd0323b20752846e6587272d3b03b0343c8ea28a6f819e6e71f" +dependencies = [ + "async-trait", + "axum-core", + "bytes", + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-util", + "itoa", + "matchit", + "memchr", + "mime", + "percent-encoding", + "pin-project-lite", + "rustversion", + "serde", + "serde_json", + "serde_path_to_error", + "serde_urlencoded", + "sync_wrapper", + "tokio", + "tower", + "tower-layer", + "tower-service", +] + +[[package]] +name = "axum-core" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09f2bd6146b97ae3359fa0cc6d6b376d9539582c7b4220f041a33ec24c226199" +dependencies = [ + "async-trait", + "bytes", + "futures-util", + "http", + "http-body", + "http-body-util", + "mime", + "pin-project-lite", + "rustversion", + "sync_wrapper", + "tower-layer", + "tower-service", +] + +[[package]] +name = "base16ct" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd307490d624467aa6f74b0eabb77633d1f758a7b25f12bceb0b22e08d9726f6" + +[[package]] +name = "base64" +version = "0.21.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" + +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + +[[package]] +name = "base64ct" +version = "1.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2af50177e190e07a26ab74f8b1efbfe2ef87da2116221318cb1c2e82baf7de06" + +[[package]] +name = "bcrypt-pbkdf" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "144e573728da132683b9488acd528274c790e07fc06ff81ee29f9d8f8b1041e0" +dependencies = [ + "blowfish", + "pbkdf2", + "sha2 0.11.0", +] + +[[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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4512299f36f043ab09a583e57bceb5a5aab7a73db1805848e8fef3c9e8c78b3" +dependencies = [ + "serde_core", +] + +[[package]] +name = "blake2" +version = "0.11.0-rc.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "061f1a09225e328e1ffbb378d2d49923c0ca5fee19fb5ac1cc9c1e9d52b93690" +dependencies = [ + "digest 0.11.3", +] + +[[package]] +name = "block-buffer" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" +dependencies = [ + "generic-array 0.14.7", +] + +[[package]] +name = "block-buffer" +version = "0.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cdd35008169921d80bc60d3d0ab416eecb028c4cd653352907921d95084790be" +dependencies = [ + "hybrid-array", + "zeroize", +] + +[[package]] +name = "block-padding" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8894febbff9f758034a5b8e12d87918f56dfc64a8e1fe757d65e29041538d93" +dependencies = [ + "generic-array 0.14.7", +] + +[[package]] +name = "block-padding" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "710f1dd022ef4e93f8a438b4ba958de7f64308434fa6a87104481645cc30068b" +dependencies = [ + "hybrid-array", +] + +[[package]] +name = "block2" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cdeb9d870516001442e364c5220d3574d2da8dc765554b4a617230d33fa58ef5" +dependencies = [ + "objc2", +] + +[[package]] +name = "blowfish" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62ce3946557b35e71d1bbe07ec385073ce9eda05043f95de134eb578fcf1a298" +dependencies = [ + "byteorder", + "cipher", +] + +[[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 = "bs58" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf88ba1141d185c399bee5288d850d63b8369520c1eafc32a0430b5b6c287bf4" +dependencies = [ + "tinyvec", +] + +[[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.1", + "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 = "cbc" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce2dc9ee5f88d11e0beb842c88b33c8a5cf0d1329c4b19494af42b07dbfe8896" +dependencies = [ + "cipher", +] + +[[package]] +name = "cc" +version = "1.2.62" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1dce859f0832a7d088c4f1119888ab94ef4b5d6795d1ce05afb7fe159d79f98" +dependencies = [ + "find-msvc-tools", + "shlex", +] + +[[package]] +name = "cesu8" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6d43a04d8753f35258c91f8ec639f792891f748a1edbd759cf1dcea3382ad83c" + +[[package]] +name = "cfb" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d38f2da7a0a2c4ccf0065be06397cc26a81f4e528be095826eee9d4adbb8c60f" +dependencies = [ + "byteorder", + "fnv", + "uuid", +] + +[[package]] +name = "cfg-expr" +version = "0.15.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d067ad48b8650848b989a59a86c6c36a995d02d2bf778d45c3c5d57bc2718f02" +dependencies = [ + "smallvec", + "target-lexicon", +] + +[[package]] +name = "cfg-if" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9330f8b2ff13f34540b44e946ef35111825727b38d33286ef986142615121801" + +[[package]] +name = "cfg_aliases" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724" + +[[package]] +name = "chacha20" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6f8d983286843e49675a4b7a2d174efe136dc93a18d69130dd18198a6c167601" +dependencies = [ + "cfg-if", + "cipher", + "cpufeatures 0.3.0", + "rand_core", + "zeroize", +] + +[[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 = "chrono-tz" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a6139a8597ed92cf816dfb33f5dd6cf0bb93a6adc938f11039f371bc5bcd26c3" +dependencies = [ + "chrono", + "phf 0.12.1", +] + +[[package]] +name = "cipher" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8cf2a2c93cd704877c0858356ed03480ff301ee950b43f1cbe4573b088bfa6c" +dependencies = [ + "block-buffer 0.12.0", + "crypto-common 0.2.2", + "inout 0.2.2", + "zeroize", +] + +[[package]] +name = "cmov" +version = "0.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f88a43d011fc4a6876cb7344703e297c71dda42494fee094d5f7c76bf13f746" + +[[package]] +name = "combine" +version = "4.6.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd" +dependencies = [ + "bytes", + "memchr", +] + +[[package]] +name = "const-oid" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a6ef517f0926dd24a1582492c791b6a4818a4d94e789a334894aa15b0d12f55c" + +[[package]] +name = "cookie" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ddef33a339a91ea89fb53151bd0a4689cfce27055c291dfa69945475d22c747" +dependencies = [ + "time", + "version_check", +] + +[[package]] +name = "core-foundation" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2a6cd9ae233e7f62ba4e9353e81a88df7fc8a5987b8d445b4d90c879bd156f6" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "core-foundation-sys" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b" + +[[package]] +name = "core-graphics" +version = "0.25.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "064badf302c3194842cf2c5d61f56cc88e54a759313879cdf03abdd27d0c3b97" +dependencies = [ + "bitflags 2.11.1", + "core-foundation", + "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.1", + "core-foundation", + "libc", +] + +[[package]] +name = "cpubits" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15b85f9c39137c3a891689859392b1bd49812121d0d61c9caf00d46ed5ce06ae" + +[[package]] +name = "cpufeatures" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280" +dependencies = [ + "libc", +] + +[[package]] +name = "cpufeatures" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b2a41393f66f16b0823bb79094d54ac5fbd34ab292ddafb9a0456ac9f87d201" +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 = "cron" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6f8c3e73077b4b4a6ab1ea5047c37c57aee77657bc8ecd6f29b0af082d0b0c07" +dependencies = [ + "chrono", + "nom", + "once_cell", +] + +[[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-bigint" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42a0d26b245348befa0c121944541476763dcc46ede886c88f9d12e1697d27c3" +dependencies = [ + "cpubits", + "ctutils", + "getrandom 0.4.2", + "hybrid-array", + "num-traits", + "rand_core", + "serdect", + "subtle", + "zeroize", +] + +[[package]] +name = "crypto-common" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78c8292055d1c1df0cce5d180393dc8cce0abec0a7102adb6c7b1eef6016d60a" +dependencies = [ + "generic-array 0.14.7", + "typenum", +] + +[[package]] +name = "crypto-common" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce6e4c961d6cd6c9a86db418387425e8bdeaf05b3c8bc1411e6dca4c252f1453" +dependencies = [ + "getrandom 0.4.2", + "hybrid-array", + "rand_core", +] + +[[package]] +name = "crypto-primes" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21f41f23de7d24cdbda7f0c4d9c0351f99a4ceb258ef30e5c1927af8987ffe5a" +dependencies = [ + "crypto-bigint", + "libm", + "rand_core", +] + +[[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.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "352d39c2f7bef1d6ad73db6f5160efcaed66d94ef8c6c573a8410c00bf909a98" +dependencies = [ + "ctor-proc-macro", + "dtor", +] + +[[package]] +name = "ctor-proc-macro" +version = "0.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52560adf09603e58c9a7ee1fe1dcb95a16927b17c127f0ac02d6e768a0e25bc1" + +[[package]] +name = "ctr" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baaca1c4b237092596f64d571e9db6ce4109c4ef9742e27590f1709594461f21" +dependencies = [ + "cipher", +] + +[[package]] +name = "ctutils" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7d5515a3834141de9eafb9717ad39eea8247b5674e6066c404e8c4b365d2a29e" +dependencies = [ + "cmov", + "subtle", +] + +[[package]] +name = "curve25519-dalek" +version = "5.0.0-pre.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "335f1947f241137a14106b6f5acc5918a5ede29c9d71d3f2cb1678d5075d9fc3" +dependencies = [ + "cfg-if", + "cpufeatures 0.2.17", + "curve25519-dalek-derive", + "digest 0.11.3", + "fiat-crypto", + "rustc_version", + "subtle", + "zeroize", +] + +[[package]] +name = "curve25519-dalek-derive" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f46882e17999c6cc590af592290432be3bce0428cb0d5f8b6715e4dc7b383eb3" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[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 = "data-encoding" +version = "2.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4ae5f15dda3c708c0ade84bfee31ccab44a3da4f88015ed22f63732abe300c8" + +[[package]] +name = "dbus" +version = "0.9.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b942602992bb7acfd1f51c49811c58a610ef9181b6e66f3e519d79b540a3bf73" +dependencies = [ + "libc", + "libdbus-sys", + "windows-sys 0.61.2", +] + +[[package]] +name = "delegate" +version = "0.13.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "780eb241654bf097afb00fc5f054a09b687dad862e485fdcf8399bb056565370" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "der" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "71fd89660b2dc699704064e59e9dba0147b903e85319429e131620d022be411b" +dependencies = [ + "const-oid", + "pem-rfc7468", + "zeroize", +] + +[[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_arbitrary" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e567bd82dcff979e4b03460c307b3cdc9e96fde3d73bed1496d2bc75d9dd62a" +dependencies = [ + "proc-macro2", + "quote", + "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 = "des" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "916a94e407b54f9034d71dd748234cd1e516ced6284009906ae246f177eafe5a" +dependencies = [ + "cipher", +] + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer 0.10.4", + "crypto-common 0.1.7", +] + +[[package]] +name = "digest" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f1dd6dbb5841937940781866fa1281a1ff7bd3bf827091440879f9994983d5c2" +dependencies = [ + "block-buffer 0.12.0", + "const-oid", + "crypto-common 0.2.2", + "ctutils", +] + +[[package]] +name = "dirs" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3e8aa94d75141228480295a7d0e7feb620b1a5ad9f12bc40be62411e38cce4e" +dependencies = [ + "dirs-sys", +] + +[[package]] +name = "dirs-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e01a3366d27ee9890022452ee61b2b63a67e6f13f58900b651ff5665f0bb1fab" +dependencies = [ + "libc", + "option-ext", + "redox_users", + "windows-sys 0.61.2", +] + +[[package]] +name = "dispatch2" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e0e367e4e7da84520dedcac1901e4da967309406d1e51017ae1abfb97adbd38" +dependencies = [ + "bitflags 2.11.1", + "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", + "foldhash 0.2.0", + "html5ever", + "precomputed-hash", + "selectors", + "tendril", +] + +[[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 = "dtor" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f1057d6c64987086ff8ed0fd3fbf377a6b7d205cc7715868cd401705f715cbe4" +dependencies = [ + "dtor-proc-macro", +] + +[[package]] +name = "dtor-proc-macro" +version = "0.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f678cf4a922c215c63e0de95eb1ff08a958a81d47e485cf9da1e27bf6305cfa5" + +[[package]] +name = "dunce" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92773504d58c093f6de2459af4af33faa518c13451eb8f2b5698ed3d36e7c813" + +[[package]] +name = "dune-dedicated-server-manager-app" +version = "0.2.0" +dependencies = [ + "base64 0.22.1", + "chrono", + "dune-manager-core", + "reqwest 0.12.28", + "serde", + "serde_json", + "tauri", + "tauri-build", + "tauri-plugin-dialog", + "tauri-plugin-process", + "tauri-plugin-shell", + "tauri-plugin-updater", +] + +[[package]] +name = "dune-manager-core" +version = "0.1.0" +dependencies = [ + "native-tls", + "postgres", + "reqwest 0.12.28", + "russh", + "serde", + "serde_json", + "sha2 0.10.9", + "tokio", + "url", +] + +[[package]] +name = "dune-server-service" +version = "0.3.16" +dependencies = [ + "anyhow", + "async-trait", + "axum", + "base64 0.22.1", + "chrono", + "chrono-tz", + "cron", + "futures", + "once_cell", + "regex", + "rusqlite", + "serde", + "serde_json", + "thiserror 1.0.69", + "tokio", + "tokio-postgres", + "tokio-util", + "tower-http", + "tracing", + "tracing-subscriber", +] + +[[package]] +name = "dyn-clone" +version = "1.0.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d0881ea181b1df73ff77ffaaf9c7544ecc11e82fba9b5f27b262a3c73a332555" + +[[package]] +name = "ecdsa" +version = "0.17.0-rc.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "54fb064faabbee66e1fc8e5c5a9458d4269dc2d8b638fe86a425adb2510d1a96" +dependencies = [ + "der", + "digest 0.11.3", + "elliptic-curve", + "rfc6979", + "signature", + "spki", + "zeroize", +] + +[[package]] +name = "ed25519" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29fcf32e6c73d1079f83ab4d782de2d81620346a5f38c6237a86a22f8368980a" +dependencies = [ + "pkcs8", + "signature", +] + +[[package]] +name = "ed25519-dalek" +version = "3.0.0-pre.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20449acd54b660981ae5caa2bcb56d1fe7f25f2e37a38ec507400fab034d4bb6" +dependencies = [ + "curve25519-dalek", + "ed25519", + "rand_core", + "serde", + "sha2 0.11.0", + "signature", + "subtle", + "zeroize", +] + +[[package]] +name = "elliptic-curve" +version = "0.14.0-rc.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cda94f31325c4275e9706adecbb6f0650dee2f904c915a98e3d81adaaaa757aa" +dependencies = [ + "base16ct", + "crypto-bigint", + "crypto-common 0.2.2", + "digest 0.11.3", + "hkdf", + "hybrid-array", + "once_cell", + "pem-rfc7468", + "pkcs8", + "rand_core", + "rustcrypto-ff", + "rustcrypto-group", + "sec1", + "subtle", + "zeroize", +] + +[[package]] +name = "embed-resource" +version = "3.0.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c31a88c8d26de40ed18fe748c547845aa39de1db3afd958f8cb91579f3644bcb" +dependencies = [ + "cc", + "memchr", + "rustc_version", + "toml 1.1.2+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 = "enum_dispatch" +version = "0.3.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa18ce2bc66555b3218614519ac839ddb759a7d6720732f979ef8d13be147ecd" +dependencies = [ + "once_cell", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[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.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4443176a9f2c162692bd3d352d745ef9413eec5782a80d8fd6f8a1ac692a07f7" + +[[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 = "fiat-crypto" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "64cd1e32ddd350061ae6edb1b082d7c54915b5c672c389143b9a63403a109f24" + +[[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 = "filetime" +version = "0.2.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2d5b2eef6fafbf69f877e55509ce5b11a760690ac9700a2921be067aa6afaef6" +dependencies = [ + "cfg-if", + "libc", +] + +[[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 = "futures" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b147ee9d1f6d097cef9ce628cd2ee62288d963e16fb287bd9286455b241382d" +dependencies = [ + "futures-channel", + "futures-core", + "futures-executor", + "futures-io", + "futures-sink", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-channel" +version = "0.3.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07bbe89c50d7a535e539b8c17bc0b49bdb77747034daa8087407d655f3f7cc1d" +dependencies = [ + "futures-core", + "futures-sink", +] + +[[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-channel", + "futures-core", + "futures-io", + "futures-macro", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "slab", +] + +[[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 = "generic-array" +version = "1.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dab9e9188e97a93276e1fe7b56401b851e2b45a46d045ca658100c1303ada649" +dependencies = [ + "generic-array 0.14.7", + "rustversion", + "typenum", +] + +[[package]] +name = "getrandom" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff2abc00be7fca6ebc474524697ae276ad847ad0a6b3faa4bcb027e9a4614ad0" +dependencies = [ + "cfg-if", + "js-sys", + "libc", + "wasi 0.11.1+wasi-snapshot-preview1", + "wasm-bindgen", +] + +[[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", + "rand_core", + "wasip2", + "wasip3", +] + +[[package]] +name = "ghash" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2eecf2d5dc9b66b732b97707a0210906b1d30523eb773193ab777c0c84b3e8d5" +dependencies = [ + "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.1", + "futures-channel", + "futures-core", + "futures-executor", + "futures-task", + "futures-util", + "gio-sys", + "glib-macros", + "glib-sys", + "gobject-sys", + "libc", + "memchr", + "once_cell", + "smallvec", + "thiserror 1.0.69", +] + +[[package]] +name = "glib-macros" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bb0228f477c0900c880fd78c8759b95c7636dbd7842707f49e132378aa2acdc" +dependencies = [ + "heck 0.4.1", + "proc-macro-crate 2.0.2", + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "glib-sys" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "063ce2eb6a8d0ea93d2bf8ba1957e78dbab6be1c2220dd3daca57d5a9d869898" +dependencies = [ + "libc", + "system-deps", +] + +[[package]] +name = "glob" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0cc23270f6e1808e30a928bdc84dea0b9b4136a8bc82338574f23baf47bbd280" + +[[package]] +name = "gobject-sys" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0850127b514d1c4a4654ead6dedadb18198999985908e6ffe4436f53c785ce44" +dependencies = [ + "glib-sys", + "libc", + "system-deps", +] + +[[package]] +name = "gtk" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd56fb197bfc42bd5d2751f4f017d44ff59fbb58140c6b49f9b3b2bdab08506a" +dependencies = [ + "atk", + "cairo-rs", + "field-offset", + "futures-channel", + "gdk", + "gdk-pixbuf", + "gio", + "glib", + "gtk-sys", + "gtk3-macros", + "libc", + "pango", + "pkg-config", +] + +[[package]] +name = "gtk-sys" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f29a1c21c59553eb7dd40e918be54dccd60c52b049b75119d5d96ce6b624414" +dependencies = [ + "atk-sys", + "cairo-sys-rs", + "gdk-pixbuf-sys", + "gdk-sys", + "gio-sys", + "glib-sys", + "gobject-sys", + "libc", + "pango-sys", + "system-deps", +] + +[[package]] +name = "gtk3-macros" +version = "0.18.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52ff3c5b21f14f0736fed6dcfc0bfb4225ebf5725f3c0209edeec181e4d73e9d" +dependencies = [ + "proc-macro-crate 1.3.1", + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "hashbrown" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" + +[[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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed5909b6e89a2db4456e54cd5f673791d7eca6732202bbf2a9cc504fe2f9b84a" + +[[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 = "hex-literal" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e712f64ec3850b98572bffac52e2c6f282b29fe6c5fa6d42334b30be438d95c1" + +[[package]] +name = "hkdf" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4aaa26c720c68b866f2c96ef5c1264b3e6f473fe5d4ce61cd44bbe913e553018" +dependencies = [ + "hmac", +] + +[[package]] +name = "hmac" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6303bc9732ae41b04cb554b844a762b4115a61bfaa81e3e83050991eeb56863f" +dependencies = [ + "digest 0.11.3", +] + +[[package]] +name = "html5ever" +version = "0.38.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1054432bae2f14e0061e33d23402fbaa67a921d319d56adc6bcf887ddad1cbc2" +dependencies = [ + "log", + "markup5ever", +] + +[[package]] +name = "http" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3ba2a386d7f85a81f119ad7498ebe444d2e22c2af0b86b069416ace48b3311a" +dependencies = [ + "bytes", + "itoa", +] + +[[package]] +name = "http-body" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" +dependencies = [ + "bytes", + "http", +] + +[[package]] +name = "http-body-util" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b021d93e26becf5dc7e1b75b1bed1fd93124b374ceb73f43d4d4eafec896a64a" +dependencies = [ + "bytes", + "futures-core", + "http", + "http-body", + "pin-project-lite", +] + +[[package]] +name = "httparse" +version = "1.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6dbf3de79e51f3d586ab4cb9d5c3e2c14aa28ed23d180cf89b4df0454a69cc87" + +[[package]] +name = "httpdate" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9" + +[[package]] +name = "hybrid-array" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9155a582abd142abc056962c29e3ce5ff2ad5469f4246b537ed42c5deba857da" +dependencies = [ + "ctutils", + "subtle", + "typenum", + "zeroize", +] + +[[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", + "http", + "http-body", + "httparse", + "httpdate", + "itoa", + "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-rustls" +version = "0.27.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "33ca68d021ef39cf6463ab54c1d0f5daf03377b70561305bb89a8f83aab66e0f" +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", + "tokio", + "tower-service", + "tracing", +] + +[[package]] +name = "iana-time-zone" +version = "0.1.65" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e31bc9ad994ba00e440a8aa5c9ef0ec67d5cb5e5cb0cc7f8b744a35b389cc470" +dependencies = [ + "android_system_properties", + "core-foundation-sys", + "iana-time-zone-haiku", + "js-sys", + "log", + "wasm-bindgen", + "windows-core 0.62.2", +] + +[[package]] +name = "iana-time-zone-haiku" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f" +dependencies = [ + "cc", +] + +[[package]] +name = "ico" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3e795dff5605e0f04bff85ca41b51a96b83e80b281e96231bcaaf1ac35103371" +dependencies = [ + "byteorder", + "png 0.17.16", +] + +[[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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb68373c0d6620ef8105e855e7745e18b0d00d3bdb07fb532e434244cdb9a714" +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.1", + "serde", + "serde_core", +] + +[[package]] +name = "infer" +version = "0.19.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a588916bfdfd92e71cacef98a63d9b1f0d74d6599980d11894290e7ddefffcf7" +dependencies = [ + "cfb", +] + +[[package]] +name = "inout" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "879f10e63c20629ecabbb64a8010319738c66a5cd0c29b02d63d272b03751d01" +dependencies = [ + "block-padding 0.3.3", + "generic-array 0.14.7", +] + +[[package]] +name = "inout" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4250ce6452e92010fdf7268ccc5d14faa80bb12fc741938534c58f16804e03c7" +dependencies = [ + "block-padding 0.4.2", + "hybrid-array", +] + +[[package]] +name = "internal-russh-num-bigint" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ae8e22120c32fb4d19ec55fba35015f57095cd95a2e3b732e44457f5915b2ee8" +dependencies = [ + "num-integer", + "num-traits", + "rand", + "rand_core", +] + +[[package]] +name = "ipnet" +version = "2.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d98f6fed1fde3f8c21bc40a1abb88dd75e67924f9cffc3ef95607bad8017f8e2" + +[[package]] +name = "is-docker" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "928bae27f42bc99b60d9ac7334e3a21d10ad8f1835a4e12ec3ec0464765ed1b3" +dependencies = [ + "once_cell", +] + +[[package]] +name = "is-wsl" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "173609498df190136aa7dea1a91db051746d339e18476eed5ca40521f02d7aa5" +dependencies = [ + "is-docker", + "once_cell", +] + +[[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" +version = "0.22.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5efd9a482cf3a427f00d6b35f14332adc7902ce91efb778580e180ff90fa3498" +dependencies = [ + "cfg-if", + "combine", + "jni-macros", + "jni-sys 0.4.1", + "log", + "simd_cesu8", + "thiserror 2.0.18", + "walkdir", + "windows-link 0.2.1", +] + +[[package]] +name = "jni-macros" +version = "0.22.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a00109accc170f0bdb141fed3e393c565b6f5e072365c3bd58f5b062591560a3" +dependencies = [ + "proc-macro2", + "quote", + "rustc_version", + "simd_cesu8", + "syn 2.0.117", +] + +[[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.98" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67df7112613f8bfd9150013a0314e196f4800d3201ae742489d999db2f979f08" +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 = "keccak" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e24a010dd405bd7ed803e5253182815b41bf2e6a80cc3bfc066658e03a198aa" +dependencies = [ + "cfg-if", + "cpufeatures 0.3.0", +] + +[[package]] +name = "kem" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "01737161ba802849cfd486b5bd209d38ba4943494c249a8126005170c7621edd" +dependencies = [ + "crypto-common 0.2.2", + "rand_core", +] + +[[package]] +name = "keyboard-types" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b750dcadc39a09dbadd74e118f6dd6598df77fa01df0cfcdc52c28dece74528a" +dependencies = [ + "bitflags 2.11.1", + "serde", + "unicode-segmentation", +] + +[[package]] +name = "lazy_static" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe" + +[[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.186" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68ab91017fe16c622486840e4c83c9a37afeff978bd239b5293d61ece587de66" + +[[package]] +name = "libdbus-sys" +version = "0.2.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "328c4789d42200f1eeec05bd86c9c13c7f091d2ba9a6ea35acdf51f31bc0f043" +dependencies = [ + "pkg-config", +] + +[[package]] +name = "libloading" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b67380fd3b2fbe7527a606e18729d21c6f3951633d0500574c4dc22d2d638b9f" +dependencies = [ + "cfg-if", + "winapi", +] + +[[package]] +name = "libm" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6d2cec3eae94f9f509c767b45932f1ada8350c4bdb85af2fcab4a3c14807981" + +[[package]] +name = "libredox" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e02f3bb43d335493c96bf3fd3a321600bf6bd07ed34bc64118e9293bdffea46c" +dependencies = [ + "libc", +] + +[[package]] +name = "libsqlite3-sys" +version = "0.30.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e99fb7a497b1e3339bc746195567ed8d3e24945ecd636e3619d20b9de9e9149" +dependencies = [ + "cc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "linux-raw-sys" +version = "0.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 = "markup5ever" +version = "0.38.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8983d30f2915feeaaab2d6babdd6bc7e9ed1a00b66b5e6d74df19aa9c0e91862" +dependencies = [ + "log", + "tendril", + "web_atoms", +] + +[[package]] +name = "matchers" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d1525a2a28c7f4fa0fc98bb91ae755d1e2d1505079e05539e35bc876b5d65ae9" +dependencies = [ + "regex-automata", +] + +[[package]] +name = "matchit" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e7465ac9959cc2b1404e8e2367b43684a6d13790fe23056cc8c6c5a6b7bcb94" + +[[package]] +name = "md-5" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69b6441f590336821bb897fb28fc622898ccceb1d6cea3fde5ea86b090c4de98" +dependencies = [ + "cfg-if", + "digest 0.11.3", +] + +[[package]] +name = "md5" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "490cc448043f947bae3cbee9c203358d62dbee0db12107a74be5c30ccfd09771" + +[[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 = "mime_guess" +version = "2.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f7c44f8e672c00fe5308fa235f821cb4198414e1c77935c1ab6948d3fd78550e" +dependencies = [ + "mime", + "unicase", +] + +[[package]] +name = "minimal-lexical" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68354c5c6bd36d73ff3feceb05efa59b6acb7626617f4962be322a825e61f79a" + +[[package]] +name = "minisign-verify" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22f9645cb765ea72b8111f36c522475d2daa0d22c957a9826437e97534bc4e9e" + +[[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 = "ml-kem" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e15f3e5b957493873e396a66914e83e616b6afe335cdef7efe5c6e1216aba66" +dependencies = [ + "hybrid-array", + "kem", + "module-lattice", + "pkcs8", + "rand_core", + "sha3", +] + +[[package]] +name = "module-lattice" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c61b87c9683ab7cb1c6871d261ad5479b6b10ceb52c4352aaca3b5d35a8febe" +dependencies = [ + "ctutils", + "hybrid-array", + "num-traits", +] + +[[package]] +name = "muda" +version = "0.19.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ae8844f63b5b118e334e205585b8c5c17b984121dbdb179d44aeb087ffad3cb" +dependencies = [ + "crossbeam-channel", + "dpi", + "gtk", + "keyboard-types", + "objc2", + "objc2-app-kit", + "objc2-core-foundation", + "objc2-foundation", + "once_cell", + "png 0.18.1", + "serde", + "thiserror 2.0.18", + "windows-sys 0.61.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.1", + "jni-sys 0.3.1", + "log", + "ndk-sys", + "num_enum", + "raw-window-handle", + "thiserror 1.0.69", +] + +[[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 = "nix" +version = "0.31.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf20d2fde8ff38632c426f1165ed7436270b44f199fc55284c38276f9db47c3d" +dependencies = [ + "bitflags 2.11.1", + "cfg-if", + "cfg_aliases", + "libc", +] + +[[package]] +name = "nom" +version = "7.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d273983c5a657a70a3e8f2a01329822f3b8c8172b73826411a55751e404a0a4a" +dependencies = [ + "memchr", + "minimal-lexical", +] + +[[package]] +name = "nu-ansi-term" +version = "0.50.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7957b9740744892f114936ab4a57b3f487491bbeafaf8083688b16841a4240e5" +dependencies = [ + "windows-sys 0.61.2", +] + +[[package]] +name = "num-bigint" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a5e44f723f1133c9deac646763579fdb3ac745e418f2a7af9cd0c431da1f20b9" +dependencies = [ + "num-integer", + "num-traits", +] + +[[package]] +name = "num-conv" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6673768db2d862beb9b39a78fdcb1a69439615d5794a1be50caa9bc92c81967" + +[[package]] +name = "num-integer" +version = "0.1.46" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f" +dependencies = [ + "num-traits", +] + +[[package]] +name = "num-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.1", + "block2", + "objc2", + "objc2-core-foundation", + "objc2-foundation", +] + +[[package]] +name = "objc2-cloud-kit" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73ad74d880bb43877038da939b7427bba67e9dd42004a18b809ba7d87cee241c" +dependencies = [ + "bitflags 2.11.1", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-data" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b402a653efbb5e82ce4df10683b6b28027616a2715e90009947d50b8dd298fa" +dependencies = [ + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-foundation" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2a180dd8642fa45cdb7dd721cd4c11b1cadd4929ce112ebd8b9f5803cc79d536" +dependencies = [ + "bitflags 2.11.1", + "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.1", + "dispatch2", + "objc2", + "objc2-core-foundation", + "objc2-io-surface", +] + +[[package]] +name = "objc2-core-image" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5d563b38d2b97209f8e861173de434bd0214cf020e3423a52624cd1d989f006" +dependencies = [ + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-location" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca347214e24bc973fc025fd0d36ebb179ff30536ed1f80252706db19ee452009" +dependencies = [ + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-text" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0cde0dfb48d25d2b4862161a4d5fcc0e3c24367869ad306b0c9ec0073bfed92d" +dependencies = [ + "bitflags 2.11.1", + "objc2", + "objc2-core-foundation", + "objc2-core-graphics", +] + +[[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.1", + "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.1", + "objc2", + "objc2-core-foundation", +] + +[[package]] +name = "objc2-osa-kit" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f112d1746737b0da274ef79a23aac283376f335f4095a083a267a082f21db0c0" +dependencies = [ + "bitflags 2.11.1", + "objc2", + "objc2-app-kit", + "objc2-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.1", + "objc2", + "objc2-core-foundation", + "objc2-foundation", +] + +[[package]] +name = "objc2-system-configuration" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7216bd11cbda54ccabcab84d523dc93b858ec75ecfb3a7d89513fa22464da396" +dependencies = [ + "objc2-core-foundation", +] + +[[package]] +name = "objc2-ui-kit" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d87d638e33c06f577498cbcc50491496a3ed4246998a7fbba7ccb98b1e7eab22" +dependencies = [ + "bitflags 2.11.1", + "block2", + "objc2", + "objc2-cloud-kit", + "objc2-core-data", + "objc2-core-foundation", + "objc2-core-graphics", + "objc2-core-image", + "objc2-core-location", + "objc2-core-text", + "objc2-foundation", + "objc2-quartz-core", + "objc2-user-notifications", +] + +[[package]] +name = "objc2-user-notifications" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9df9128cbbfef73cda168416ccf7f837b62737d748333bfe9ab71c245d76613e" +dependencies = [ + "objc2", + "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.1", + "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 = "open" +version = "5.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2fbaa89d2ddc8473c78a3adf69eea8cffa28c483b8e02a971ef31527cd0fc92c" +dependencies = [ + "dunce", + "is-wsl", + "libc", + "pathdiff", +] + +[[package]] +name = "openssl" +version = "0.10.79" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf0b434746ee2832f4f0baf10137e1cabb18cbe6912c69e2e33263c45250f542" +dependencies = [ + "bitflags 2.11.1", + "cfg-if", + "foreign-types 0.3.2", + "libc", + "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.115" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "158fe5b292746440aa6e7a7e690e55aeb72d41505e2804c23c6973ad0e9c9781" +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 = "os_pipe" +version = "1.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7d8fae84b431384b68627d0f9b3b1245fcf9f46f6c0e3dc902e9dce64edd1967" +dependencies = [ + "libc", + "windows-sys 0.61.2", +] + +[[package]] +name = "osakit" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "732c71caeaa72c065bb69d7ea08717bd3f4863a4f451402fc9513e29dbd5261b" +dependencies = [ + "objc2", + "objc2-foundation", + "objc2-osa-kit", + "serde", + "serde_json", + "thiserror 2.0.18", +] + +[[package]] +name = "p256" +version = "0.14.0-rc.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b97e3bf0465157ae90975ff52dbeb1362ba618924878c9f74c25baa27a65f9a" +dependencies = [ + "ecdsa", + "elliptic-curve", + "primefield", + "primeorder", + "sha2 0.11.0", +] + +[[package]] +name = "p384" +version = "0.14.0-rc.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "437f30ebcb1e16ff48acead5f08bd69fbcdbc82421687bb48af5c315a0bfab03" +dependencies = [ + "ecdsa", + "elliptic-curve", + "fiat-crypto", + "primefield", + "primeorder", + "sha2 0.11.0", +] + +[[package]] +name = "p521" +version = "0.14.0-rc.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4e9fd792bab86ecf6249561752fb5a413511f999887107dd054bbda5143743d7" +dependencies = [ + "base16ct", + "ecdsa", + "elliptic-curve", + "primefield", + "primeorder", + "sha2 0.11.0", +] + +[[package]] +name = "pageant" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f3a5ae18f65a85c67a77d18d42d3606c07948e3c17c1e5f74852b26589e88a5" +dependencies = [ + "base16ct", + "byteorder", + "bytes", + "delegate", + "futures", + "log", + "rand", + "sha2 0.11.0", + "thiserror 2.0.18", + "tokio", + "windows 0.62.2", + "windows-strings 0.5.1", +] + +[[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 = "password-hash" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aab41826031698d6ffcd9cff78ef56ef998e39dc7e5067cdfebe373842d4723b" +dependencies = [ + "phc", +] + +[[package]] +name = "pathdiff" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df94ce210e5bc13cb6651479fa48d14f601d9858cfe0467f43ae157023b938d3" + +[[package]] +name = "pbkdf2" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "112d82ceb8c5bf524d9af484d4e4970c9fd5a0cc15ba14ad93dccd28873b0629" +dependencies = [ + "digest 0.11.3", + "hmac", +] + +[[package]] +name = "pem-rfc7468" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a6305423e0e7738146434843d1694d621cce767262b2a86910beab705e4493d9" +dependencies = [ + "base64ct", +] + +[[package]] +name = "percent-encoding" +version = "2.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b4f627cb1b25917193a259e49bdad08f671f8d9708acfd5fe0a8c1455d87220" + +[[package]] +name = "phc" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44dc769b75f93afdddd8c7fa12d685292ddeff1e66f7f0f3a234cf1818afe892" +dependencies = [ + "base64ct", + "ctutils", +] + +[[package]] +name = "phf" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "913273894cec178f401a31ec4b656318d95473527be05c0752cc41cdc32be8b7" +dependencies = [ + "phf_shared 0.12.1", +] + +[[package]] +name = "phf" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c1562dc717473dbaa4c1f85a36410e03c047b2e7df7f45ee938fbef64ae7fadf" +dependencies = [ + "phf_macros", + "phf_shared 0.13.1", + "serde", +] + +[[package]] +name = "phf_codegen" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49aa7f9d80421bca176ca8dbfebe668cc7a2684708594ec9f3c0db0805d5d6e1" +dependencies = [ + "phf_generator", + "phf_shared 0.13.1", +] + +[[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.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "812f032b54b1e759ccd5f8b6677695d5268c588701effba24601f6932f8269ef" +dependencies = [ + "phf_generator", + "phf_shared 0.13.1", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "phf_shared" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "06005508882fb681fd97892ecff4b7fd0fee13ef1aa569f8695dae7ab9099981" +dependencies = [ + "siphasher", +] + +[[package]] +name = "phf_shared" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e57fef6bc5981e38c2ce2d63bfa546861309f875b8a75f092d1d54ae2d64f266" +dependencies = [ + "siphasher", +] + +[[package]] +name = "pin-project-lite" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a89322df9ebe1c1578d689c92318e070967d1042b512afbe49518723f4e6d5cd" + +[[package]] +name = "pkcs1" +version = "0.8.0-rc.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "986d2e952779af96ea048f160fd9194e1751b4faea78bcf3ceb456efe008088e" +dependencies = [ + "der", + "spki", +] + +[[package]] +name = "pkcs5" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "279a91971a1d8eb1260a30938eae3be9cb67b472dffecb222fbbbe2fd2dc1453" +dependencies = [ + "aes", + "cbc", + "der", + "pbkdf2", + "rand_core", + "scrypt", + "sha2 0.11.0", + "spki", +] + +[[package]] +name = "pkcs8" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "451913da69c775a56034ea8d9003d27ee8948e12443eae7c038ba100a4f21cb7" +dependencies = [ + "der", + "pkcs5", + "rand_core", + "spki", +] + +[[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.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "092791278e026273c1b65bbdcfbba3a300f2994c896bd01ab01da613c29c46f1" +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 = "png" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "60769b8b31b2a9f263dae2776c37b1b28ae246943cf719eb6946a1db05128a61" +dependencies = [ + "bitflags 2.11.1", + "crc32fast", + "fdeflate", + "flate2", + "miniz_oxide", +] + +[[package]] +name = "poly1305" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a00baa632505d05512f48a963e16051c54fda9a95cc9acea1a4e3c90991c4a2e" +dependencies = [ + "cpufeatures 0.3.0", + "universal-hash", + "zeroize", +] + +[[package]] +name = "polyval" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7dfc63250416fea14f5749b90725916a6c903f599d51cb635aa7a52bfd03eede" +dependencies = [ + "cpubits", + "cpufeatures 0.3.0", + "universal-hash", +] + +[[package]] +name = "postgres" +version = "0.19.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aacf632d0554ff75f58183694f41dc8999c8a3a43a386994d0ec2d034f1dfbe1" +dependencies = [ + "bytes", + "fallible-iterator 0.2.0", + "futures-util", + "log", + "tokio", + "tokio-postgres", +] + +[[package]] +name = "postgres-protocol" +version = "0.6.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "56201207dac53e2f38e848e31b4b91616a6bb6e0c7205b77718994a7f49e70fc" +dependencies = [ + "base64 0.22.1", + "byteorder", + "bytes", + "fallible-iterator 0.2.0", + "hmac", + "md-5", + "memchr", + "rand", + "sha2 0.11.0", + "stringprep", +] + +[[package]] +name = "postgres-types" +version = "0.2.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8dc729a129e682e8d24170cd30ae1aa01b336b096cbb56df6d534ffec133d186" +dependencies = [ + "bytes", + "fallible-iterator 0.2.0", + "postgres-protocol", +] + +[[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 = "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 = "primefield" +version = "0.14.0-rc.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b52e6ee42db392378a95622b463c9740631171d1efce43fa445a569c1600cb6" +dependencies = [ + "crypto-bigint", + "crypto-common 0.2.2", + "rand_core", + "rustcrypto-ff", + "subtle", + "zeroize", +] + +[[package]] +name = "primeorder" +version = "0.14.0-rc.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0556580e42c19833f5d232aca11a7687a503ee41f937b54f5ae1d50fc2a6a36a" +dependencies = [ + "elliptic-curve", +] + +[[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-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.39.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cdcc8dd4e2f670d309a5f0e83fe36dfdc05af317008fea29144da1a2ac858e5e" +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.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d2e8e8bcc7961af1fdac401278c6a831614941f6164ee3bf4ce61b7edb162207" +dependencies = [ + "chacha20", + "getrandom 0.4.2", + "rand_core", +] + +[[package]] +name = "rand_core" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63b8176103e19a2643978565ca18b50549f6101881c443590420e4dc998a3c69" + +[[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.1", +] + +[[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", + "futures-channel", + "futures-core", + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-tls", + "hyper-util", + "js-sys", + "log", + "mime_guess", + "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.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62e0021ea2c22aed41653bc7e1419abb2c97e038ff2c33d0e1309e49a97deec0" +dependencies = [ + "base64 0.22.1", + "bytes", + "futures-core", + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-rustls", + "hyper-util", + "js-sys", + "log", + "percent-encoding", + "pin-project-lite", + "rustls", + "rustls-pki-types", + "rustls-platform-verifier", + "serde", + "serde_json", + "sync_wrapper", + "tokio", + "tokio-rustls", + "tokio-util", + "tower", + "tower-http", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-streams", + "web-sys", +] + +[[package]] +name = "rfc6979" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5236ce872cac07e0fb3969b0cbf468c7d2f37d432f1b627dcb7b8d34563fb0c3" +dependencies = [ + "hmac", + "subtle", +] + +[[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 = "rsa" +version = "0.10.0-rc.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "30b2aa4ba0d89f73d1e332df05be0eeab8840351c36ca5654341dfdb57bb3caf" +dependencies = [ + "const-oid", + "crypto-bigint", + "crypto-primes", + "digest 0.11.3", + "pkcs1", + "pkcs8", + "rand_core", + "sha2 0.11.0", + "signature", + "spki", + "zeroize", +] + +[[package]] +name = "rusqlite" +version = "0.32.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7753b721174eb8ff87a9a0e799e2d7bc3749323e773db92e0984debb00019d6e" +dependencies = [ + "bitflags 2.11.1", + "fallible-iterator 0.3.0", + "fallible-streaming-iterator", + "hashlink", + "libsqlite3-sys", + "smallvec", +] + +[[package]] +name = "russh" +version = "0.61.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fba341d1fce5e094392f088499c5808d2a60d3a658f58fe480c26ac77b1c7dd3" +dependencies = [ + "aes", + "bitflags 2.11.1", + "block-padding 0.4.2", + "byteorder", + "bytes", + "cbc", + "cipher", + "crypto-bigint", + "ctr", + "curve25519-dalek", + "data-encoding", + "delegate", + "der", + "digest 0.11.3", + "ecdsa", + "ed25519-dalek", + "elliptic-curve", + "enum_dispatch", + "flate2", + "futures", + "generic-array 1.4.1", + "getrandom 0.2.17", + "ghash", + "hex-literal", + "hkdf", + "hmac", + "inout 0.1.4", + "internal-russh-num-bigint", + "keccak", + "log", + "md5", + "ml-kem", + "module-lattice", + "num-bigint", + "p256", + "p384", + "p521", + "pageant", + "pbkdf2", + "pkcs1", + "pkcs5", + "pkcs8", + "polyval", + "rand", + "rand_core", + "ring", + "rsa", + "russh-cryptovec", + "russh-util", + "salsa20", + "scrypt", + "sec1", + "sha1", + "sha2 0.11.0", + "sha3", + "signature", + "spki", + "ssh-encoding", + "ssh-key", + "subtle", + "thiserror 2.0.18", + "tokio", + "typenum", + "universal-hash", + "zeroize", +] + +[[package]] +name = "russh-cryptovec" +version = "0.61.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "443f6bbcfacb34a1aab2b12b99bf08e0c63abdc5a0db261901365df9d57fff51" +dependencies = [ + "log", + "nix", + "ssh-encoding", + "windows-sys 0.61.2", +] + +[[package]] +name = "russh-util" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "668424a5dde0bcb45b55ba7de8476b93831b4aa2fa6947e145f3b053e22c60b6" +dependencies = [ + "chrono", + "tokio", + "wasm-bindgen", + "wasm-bindgen-futures", +] + +[[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 = "rustcrypto-ff" +version = "0.14.0-rc.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd2a8adb347447693cd2ba0d218c4b66c62da9b0a5672b17b981e4291ec65ff6" +dependencies = [ + "rand_core", + "subtle", +] + +[[package]] +name = "rustcrypto-group" +version = "0.14.0-rc.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "369f9b61aa45933c062c9f6b5c3c50ab710687eca83dd3802653b140b43f85ed" +dependencies = [ + "rand_core", + "rustcrypto-ff", + "subtle", +] + +[[package]] +name = "rustix" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6fe4565b9518b83ef4f91bb47ce29620ca828bd32cb7e408f0062e9930ba190" +dependencies = [ + "bitflags 2.11.1", + "errno", + "libc", + "linux-raw-sys", + "windows-sys 0.61.2", +] + +[[package]] +name = "rustls" +version = "0.23.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef86cd5876211988985292b91c96a8f2d298df24e75989a43a3c73f2d4d8168b" +dependencies = [ + "once_cell", + "ring", + "rustls-pki-types", + "rustls-webpki", + "subtle", + "zeroize", +] + +[[package]] +name = "rustls-native-certs" +version = "0.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "612460d5f7bea540c490b2b6395d8e34a953e52b491accd6c86c8164c5932a63" +dependencies = [ + "openssl-probe", + "rustls-pki-types", + "schannel", + "security-framework", +] + +[[package]] +name = "rustls-pki-types" +version = "1.14.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "30a7197ae7eb376e574fe940d068c30fe0462554a3ddbe4eca7838e049c937a9" +dependencies = [ + "zeroize", +] + +[[package]] +name = "rustls-platform-verifier" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26d1e2536ce4f35f4846aa13bff16bd0ff40157cdb14cc056c7b14ba41233ba0" +dependencies = [ + "core-foundation", + "core-foundation-sys", + "jni 0.22.4", + "log", + "once_cell", + "rustls", + "rustls-native-certs", + "rustls-platform-verifier-android", + "rustls-webpki", + "security-framework", + "security-framework-sys", + "webpki-root-certs", + "windows-sys 0.61.2", +] + +[[package]] +name = "rustls-platform-verifier-android" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f87165f0995f63a9fbeea62b64d10b4d9d8e78ec6d7d51fb2125fda7bb36788f" + +[[package]] +name = "rustls-webpki" +version = "0.103.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61c429a8649f110dddef65e2a5ad240f747e85f7758a6bccc7e5777bd33f756e" +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 = "salsa20" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2f874456e72520ff1375a06c588eaf074b0f01f9e9e1aada45bd9b7954a6e42c" +dependencies = [ + "cfg-if", + "cipher", +] + +[[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 = "scrypt" +version = "0.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d87af57419b594aa23fa95f09f0e06d80d84ba01c26148c43844cad6ff4485f0" +dependencies = [ + "cfg-if", + "pbkdf2", + "salsa20", + "sha2 0.11.0", +] + +[[package]] +name = "sec1" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d56d437c2f19203ce5f7122e507831de96f3d2d4d3be5af44a0b0a09d8a80e4d" +dependencies = [ + "base16ct", + "ctutils", + "der", + "hybrid-array", + "subtle", + "zeroize", +] + +[[package]] +name = "security-framework" +version = "3.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7f4bc775c73d9a02cde8bf7b2ec4c9d12743edf609006c7facc23998404cd1d" +dependencies = [ + "bitflags 2.11.1", + "core-foundation", + "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.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c5d9c0c92a92d33f08817311cf3f2c29a3538a8240e94a6a3c622ce652d7e00c" +dependencies = [ + "bitflags 2.11.1", + "cssparser", + "derive_more", + "log", + "new_debug_unreachable", + "phf 0.13.1", + "phf_codegen", + "precomputed-hash", + "rustc-hash", + "servo_arc", + "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_path_to_error" +version = "0.1.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10a9ff822e371bb5403e391ecd83e182e0e77ba7f6fe0160b795797109d1b457" +dependencies = [ + "itoa", + "serde", + "serde_core", +] + +[[package]] +name = "serde_repr" +version = "0.1.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "175ee3e80ae9982737ca543e96133087cbd9a485eecc3bc4de9c1a37b47ea59c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.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.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e72c1c2cb7b223fafb600a619537a871c2818583d619401b785e7c0b746ccde2" +dependencies = [ + "base64 0.22.1", + "bs58", + "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.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b90c488738ecb4fb0262f41f43bc40efc5868d9fb744319ddf5f5317f417bfac" +dependencies = [ + "darling", + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "serdect" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "66cf8fedced2fcf12406bcb34223dffb92eaf34908ede12fed414c82b7f00b3e" +dependencies = [ + "base16ct", + "serde", +] + +[[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.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "170fb83ab34de17dc69aa7c67482b22218ddb85da56546f9bd6b929e32a05930" +dependencies = [ + "stable_deref_trait", +] + +[[package]] +name = "sha1" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aacc4cc499359472b4abe1bf11d0b12e688af9a805fa5e3016f9a386dc2d0214" +dependencies = [ + "cfg-if", + "cpufeatures 0.3.0", + "digest 0.11.3", +] + +[[package]] +name = "sha2" +version = "0.10.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7507d819769d01a365ab707794a4084392c824f54a7a6a7862f8c3d0892b283" +dependencies = [ + "cfg-if", + "cpufeatures 0.2.17", + "digest 0.10.7", +] + +[[package]] +name = "sha2" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "446ba717509524cb3f22f17ecc096f10f4822d76ab5c0b9822c5f9c284e825f4" +dependencies = [ + "cfg-if", + "cpufeatures 0.3.0", + "digest 0.11.3", +] + +[[package]] +name = "sha3" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be176f1a57ce4e3d31c1a166222d9768de5954f811601fb7ca06fc8203905ce1" +dependencies = [ + "digest 0.11.3", + "keccak", +] + +[[package]] +name = "sharded-slab" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f40ca3c46823713e0d4209592e8d6e826aa57e928f09752619fc696c499637f6" +dependencies = [ + "lazy_static", +] + +[[package]] +name = "shared_child" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e362d9935bc50f019969e2f9ecd66786612daae13e8f277be7bfb66e8bed3f7" +dependencies = [ + "libc", + "sigchld", + "windows-sys 0.60.2", +] + +[[package]] +name = "shlex" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" + +[[package]] +name = "sigchld" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47106eded3c154e70176fc83df9737335c94ce22f821c32d17ed1db1f83badb1" +dependencies = [ + "libc", + "os_pipe", + "signal-hook", +] + +[[package]] +name = "signal-hook" +version = "0.3.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d881a16cf4426aa584979d30bd82cb33429027e42122b169753d6ef1085ed6e2" +dependencies = [ + "libc", + "signal-hook-registry", +] + +[[package]] +name = "signal-hook-registry" +version = "1.4.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4db69cba1110affc0e9f7bcd48bbf87b3f4fc7c61fc9155afd4c469eb3d6c1b" +dependencies = [ + "errno", + "libc", +] + +[[package]] +name = "signature" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "28d567dcbaf0049cb8ac2608a76cd95ff9e4412e1899d389ee400918ca7537f5" +dependencies = [ + "digest 0.11.3", + "rand_core", +] + +[[package]] +name = "simd-adler32" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "703d5c7ef118737c72f1af64ad2f6f8c5e1921f818cdcb97b8fe6fc69bf66214" + +[[package]] +name = "simd_cesu8" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94f90157bb87cddf702797c5dadfa0be7d266cdf49e22da2fcaa32eff75b2c33" +dependencies = [ + "rustc_version", + "simdutf8", +] + +[[package]] +name = "simdutf8" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3a9fe34e3e7a50316060351f37187a3f546bce95496156754b601a5fa71b76e" + +[[package]] +name = "siphasher" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ee5873ec9cce0195efcb7a4e9507a04cd49aec9c83d0389df45b1ef7ba2e649" + +[[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 = "spki" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d9efca8738c78ee9484207732f728b1ef517bbb1833d6fc0879ca898a522f6f" +dependencies = [ + "base64ct", + "der", +] + +[[package]] +name = "ssh-cipher" +version = "0.3.0-rc.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10db6f219196a8528f9ec904d9d45cdad692d65b0e57e72be4dedd1c5fddce36" +dependencies = [ + "aead", + "aes", + "aes-gcm", + "cbc", + "chacha20", + "cipher", + "ctr", + "ctutils", + "des", + "poly1305", + "ssh-encoding", + "zeroize", +] + +[[package]] +name = "ssh-encoding" +version = "0.3.0-rc.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7abf34aa716da5d5b4c496936d042ea282ab392092cd68a72ef6a8863ff8c96a" +dependencies = [ + "base64ct", + "bytes", + "crypto-bigint", + "ctutils", + "digest 0.11.3", + "pem-rfc7468", + "zeroize", +] + +[[package]] +name = "ssh-key" +version = "0.7.0-rc.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "45735ce3dea95690e4a9e414c4cfde7f79835063c3dcd35881df85a84118e74b" +dependencies = [ + "argon2", + "bcrypt-pbkdf", + "ctutils", + "ed25519-dalek", + "hex", + "hmac", + "p256", + "p384", + "p521", + "rand_core", + "rsa", + "sec1", + "sha1", + "sha2 0.11.0", + "signature", + "ssh-cipher", + "ssh-encoding", + "zeroize", +] + +[[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.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.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "585635e46db231059f76c5849798146164652513eb9e8ab2685939dd90f29b69" +dependencies = [ + "phf_generator", + "phf_shared 0.13.1", + "proc-macro2", + "quote", +] + +[[package]] +name = "stringprep" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b4df3d392d81bd458a8a621b8bffbd2302a12ffe288a9d931670948749463b1" +dependencies = [ + "unicode-bidi", + "unicode-normalization", + "unicode-properties", +] + +[[package]] +name = "strsim" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" + +[[package]] +name = "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", + "unicode-ident", +] + +[[package]] +name = "syn" +version = "2.0.117" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e665b8803e7b1d2a727f4023456bbbbe74da67099c585258af0ad9c5013b9b99" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "sync_wrapper" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0bf256ce5efdfa370213c1dabab5935a12e49f2c58d15e9eac2870d3b4f27263" +dependencies = [ + "futures-core", +] + +[[package]] +name = "synstructure" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "728a70f3dbaf5bab7f0c4b1ac8d7ae5ea60a4b5549c8a5914361c99147a709d2" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "system-deps" +version = "6.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3e535eb8dded36d55ec13eddacd30dec501792ff23a0b1682c38601b8cf2349" +dependencies = [ + "cfg-expr", + "heck 0.5.0", + "pkg-config", + "toml 0.8.2", + "version-compare", +] + +[[package]] +name = "tao" +version = "0.35.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a33f7f9e486ade65fcf1e45c440f9236c904f5c1002cdc7fc6ae582777345ce4" +dependencies = [ + "bitflags 2.11.1", + "block2", + "core-foundation", + "core-graphics", + "crossbeam-channel", + "dbus", + "dispatch2", + "dlopen2", + "dpi", + "gdkwayland-sys", + "gdkx11-sys", + "gtk", + "jni 0.21.1", + "libc", + "log", + "ndk", + "ndk-sys", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "objc2-ui-kit", + "once_cell", + "parking_lot", + "percent-encoding", + "raw-window-handle", + "tao-macros", + "unicode-segmentation", + "url", + "windows 0.61.3", + "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 = "tar" +version = "0.4.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22692a6476a21fa75fdfc11d452fda482af402c008cdbaf3476414e122040973" +dependencies = [ + "filetime", + "libc", + "xattr", +] + +[[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.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b93bd86d231f0a8138f11a02a584769fe4b703dc36ae133d783228dbc4801405" +dependencies = [ + "anyhow", + "bytes", + "cookie", + "dirs", + "dunce", + "embed_plist", + "getrandom 0.3.4", + "glob", + "gtk", + "heck 0.5.0", + "http", + "jni 0.21.1", + "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.3", + "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 0.61.3", +] + +[[package]] +name = "tauri-build" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3a318b234cc2dea65f575467bafcfb76286bce228ebc3778e337d61d03213007" +dependencies = [ + "anyhow", + "cargo_toml", + "dirs", + "glob", + "heck 0.5.0", + "json-patch", + "schemars 0.8.22", + "semver", + "serde", + "serde_json", + "tauri-utils", + "tauri-winres", + "walkdir", +] + +[[package]] +name = "tauri-codegen" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6bd11644962add2549a60b7e7c6800f17d7020156e02f516021d8103e80cc528" +dependencies = [ + "base64 0.22.1", + "brotli", + "ico", + "json-patch", + "plist", + "png 0.17.16", + "proc-macro2", + "quote", + "semver", + "serde", + "serde_json", + "sha2 0.10.9", + "syn 2.0.117", + "tauri-utils", + "thiserror 2.0.18", + "time", + "url", + "uuid", + "walkdir", +] + +[[package]] +name = "tauri-macros" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fed9d3742a37a355d2e47c9af924e9fbc112abb76f9835d35d4780e318419502" +dependencies = [ + "heck 0.5.0", + "proc-macro2", + "quote", + "syn 2.0.117", + "tauri-codegen", + "tauri-utils", +] + +[[package]] +name = "tauri-plugin" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eefb2c18e8a605c23edb48fc56bb77381199e1a1e7f6ff0c9b970afe7b3cb8ee" +dependencies = [ + "anyhow", + "glob", + "plist", + "schemars 0.8.22", + "serde", + "serde_json", + "tauri-utils", + "walkdir", +] + +[[package]] +name = "tauri-plugin-dialog" +version = "2.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "65981abb771e74e571a38196c3baa11c459379164791eba0e67abc1a5fac9884" +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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7ecc274121aca0c036a2b42d1cbe83d368d348f54e0bb8a735c2b1548e8f371" +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 1.1.2+spec-1.1.0", + "url", +] + +[[package]] +name = "tauri-plugin-process" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d55511a7bf6cd70c8767b02c97bf8134fa434daf3926cfc1be0a0f94132d165a" +dependencies = [ + "tauri", + "tauri-plugin", +] + +[[package]] +name = "tauri-plugin-shell" +version = "2.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8457dbf9e2bab1edd8df22bb2c20857a59a9868e79cb3eac5ed639eec4d0c73b" +dependencies = [ + "encoding_rs", + "log", + "open", + "os_pipe", + "regex", + "schemars 0.8.22", + "serde", + "serde_json", + "shared_child", + "tauri", + "tauri-plugin", + "thiserror 2.0.18", + "tokio", +] + +[[package]] +name = "tauri-plugin-updater" +version = "2.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "806d9dac662c2e4594ff03c647a552f2c9bd544e7d0f683ec58f872f952ce4af" +dependencies = [ + "base64 0.22.1", + "dirs", + "flate2", + "futures-util", + "http", + "infer", + "log", + "minisign-verify", + "osakit", + "percent-encoding", + "reqwest 0.13.3", + "rustls", + "semver", + "serde", + "serde_json", + "tar", + "tauri", + "tauri-plugin", + "tempfile", + "thiserror 2.0.18", + "time", + "tokio", + "url", + "windows-sys 0.60.2", + "zip", +] + +[[package]] +name = "tauri-runtime" +version = "2.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fef478ba1d2ac21c2d528740b24d0cb315e1e8b1111aae53fafac34804371fc" +dependencies = [ + "cookie", + "dpi", + "gtk", + "http", + "jni 0.21.1", + "objc2", + "objc2-ui-kit", + "objc2-web-kit", + "raw-window-handle", + "serde", + "serde_json", + "tauri-utils", + "thiserror 2.0.18", + "url", + "webkit2gtk", + "webview2-com", + "windows 0.61.3", +] + +[[package]] +name = "tauri-runtime-wry" +version = "2.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3989df2ae1c476404fe0a2e8ffc4cfbde97e51efd613c2bb5355fbc9ab52cf0" +dependencies = [ + "gtk", + "http", + "jni 0.21.1", + "log", + "objc2", + "objc2-app-kit", + "once_cell", + "percent-encoding", + "raw-window-handle", + "softbuffer", + "tao", + "tauri-runtime", + "tauri-utils", + "url", + "webkit2gtk", + "webview2-com", + "windows 0.61.3", + "wry", +] + +[[package]] +name = "tauri-utils" +version = "2.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d57200389a2f82b4b0a40ae29ca19b6978116e8f4d4e974c3234ce40c0ffbdec" +dependencies = [ + "anyhow", + "brotli", + "cargo_metadata", + "ctor", + "dom_query", + "dunce", + "glob", + "http", + "infer", + "json-patch", + "log", + "memchr", + "phf 0.13.1", + "plist", + "proc-macro2", + "quote", + "regex", + "schemars 0.8.22", + "semver", + "serde", + "serde-untagged", + "serde_json", + "serde_with", + "swift-rs", + "thiserror 2.0.18", + "toml 1.1.2+spec-1.1.0", + "url", + "urlpattern", + "uuid", + "walkdir", +] + +[[package]] +name = "tauri-winres" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc65d45c68858bfe420dd29e834b5d15dbecf8a07a8a16cf4d532c7b1f69d4b6" +dependencies = [ + "dunce", + "embed-resource", + "toml 1.1.2+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.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 = "thread_local" +version = "1.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f60246a4944f24f6e018aa17cdeffb7818b76356965d03b07d6a9886e8962185" +dependencies = [ + "cfg-if", +] + +[[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 = "tinyvec" +version = "1.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3e61e67053d25a4e82c844e8424039d9745781b3fc4f32b8d55ed50f5f667ef3" +dependencies = [ + "tinyvec_macros", +] + +[[package]] +name = "tinyvec_macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" + +[[package]] +name = "tokio" +version = "1.52.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fc7f01b389ac15039e4dc9531aa973a135d7a4135281b12d7c1bc79fd57fffe" +dependencies = [ + "bytes", + "libc", + "mio", + "pin-project-lite", + "signal-hook-registry", + "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-postgres" +version = "0.7.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4dd8df5ef180f6364759a6f00f7aadda4fbbac86cdee37480826a6ff9f3574ce" +dependencies = [ + "async-trait", + "byteorder", + "bytes", + "fallible-iterator 0.2.0", + "futures-channel", + "futures-util", + "log", + "parking_lot", + "percent-encoding", + "phf 0.13.1", + "pin-project-lite", + "postgres-protocol", + "postgres-types", + "rand", + "socket2", + "tokio", + "tokio-util", + "whoami", +] + +[[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", + "futures-util", + "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" +version = "1.1.2+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81f3d15e84cbcd896376e6730314d59fb5a87f31e4b038454184435cd57defee" +dependencies = [ + "indexmap 2.14.0", + "serde_core", + "serde_spanned 1.1.1", + "toml_datetime 1.1.1+spec-1.1.0", + "toml_parser", + "toml_writer", + "winnow 1.0.2", +] + +[[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.2", +] + +[[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.2", +] + +[[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.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68d6fdd9f81c2819c9a8b0e0cd91660e7746a8e6ea2ba7c6b2b057985f6bcb51" +dependencies = [ + "bitflags 2.11.1", + "bytes", + "futures-util", + "http", + "http-body", + "pin-project-lite", + "tower", + "tower-layer", + "tower-service", + "tracing", + "url", +] + +[[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-attributes", + "tracing-core", +] + +[[package]] +name = "tracing-attributes" +version = "0.1.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7490cfa5ec963746568740651ac6781f701c9c5ea257c58e057f3ba8cf69e8da" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.117", +] + +[[package]] +name = "tracing-core" +version = "0.1.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db97caf9d906fbde555dd62fa95ddba9eecfd14cb388e4f491a66d74cd5fb79a" +dependencies = [ + "once_cell", + "valuable", +] + +[[package]] +name = "tracing-log" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee855f1f400bd0e5c02d150ae5de3840039a3f54b025156404e34c23c03f47c3" +dependencies = [ + "log", + "once_cell", + "tracing-core", +] + +[[package]] +name = "tracing-serde" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "704b1aeb7be0d0a84fc9828cae51dab5970fee5088f83d1dd7ee6f6246fc6ff1" +dependencies = [ + "serde", + "tracing-core", +] + +[[package]] +name = "tracing-subscriber" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb7f578e5945fb242538965c2d0b04418d38ec25c79d160cd279bf0731c8d319" +dependencies = [ + "matchers", + "nu-ansi-term", + "once_cell", + "regex-automata", + "serde", + "serde_json", + "sharded-slab", + "smallvec", + "thread_local", + "tracing", + "tracing-core", + "tracing-log", + "tracing-serde", +] + +[[package]] +name = "tray-icon" +version = "0.23.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15edbb0d80583e85ee8df283410038e17314df5cba30da2087a54a85216c0773" +dependencies = [ + "crossbeam-channel", + "dirs", + "libappindicator", + "muda", + "objc2", + "objc2-app-kit", + "objc2-core-foundation", + "objc2-core-graphics", + "objc2-foundation", + "once_cell", + "png 0.18.1", + "serde", + "thiserror 2.0.18", + "windows-sys 0.61.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.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "40ce102ab67701b8526c123c1bab5cbe42d7040ccfd0f64af1a385808d2f43de" + +[[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 = "unicase" +version = "2.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dbc4bc3a9f746d862c45cb89d705aa10f187bb96c76001afab07a0d35ce60142" + +[[package]] +name = "unicode-bidi" +version = "0.3.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c1cb5db39152898a79168971543b1cb5020dff7fe43c8dc468b0885f5e29df5" + +[[package]] +name = "unicode-ident" +version = "1.0.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75" + +[[package]] +name = "unicode-normalization" +version = "0.1.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5fd4f6878c9cb28d874b009da9e8d183b5abc80117c40bbd187a1fde336be6e8" +dependencies = [ + "tinyvec", +] + +[[package]] +name = "unicode-properties" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7df058c713841ad818f1dc5d3fd88063241cc61f49f5fbea4b951e8cf5a8d71d" + +[[package]] +name = "unicode-segmentation" +version = "1.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.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4987bdc12753382e0bec4a65c50738ffaabc998b9cdd1f952fb5f39b0048a96" +dependencies = [ + "crypto-common 0.2.2", + "ctutils", +] + +[[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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ddd74a9687298c6858e9b88ec8935ec45d22e8fd5e6394fa1bd4e99a87789c76" +dependencies = [ + "getrandom 0.4.2", + "js-sys", + "serde_core", + "wasm-bindgen", +] + +[[package]] +name = "valuable" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba73ea9cf16a25df0c8caa16c51acb937d5712a8429db78a3ee29d5dcacd3a65" + +[[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.11.1+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ccf3ec651a847eb01de73ccad15eb7d99f80485de043efb2f370cd654f4ea44b" + +[[package]] +name = "wasi" +version = "0.14.7+wasi-0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "883478de20367e224c0090af9cf5f9fa85bed63a95c1abf3afc5c083ebc06e8c" +dependencies = [ + "wasip2", +] + +[[package]] +name = "wasip2" +version = "1.0.3+wasi-0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20064672db26d7cdc89c7798c48a0fdfac8213434a1186e5ef29fd560ae223d6" +dependencies = [ + "wit-bindgen 0.57.1", +] + +[[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 0.51.0", +] + +[[package]] +name = "wasite" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "66fe902b4a6b8028a753d5424909b764ccf79b7a209eac9bf97e59cda9f71a42" +dependencies = [ + "wasi 0.14.7+wasi-0.2.4", +] + +[[package]] +name = "wasm-bindgen" +version = "0.2.121" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49ace1d07c165b0864824eee619580c4689389afa9dc9ed3a4c75040d82e6790" +dependencies = [ + "cfg-if", + "once_cell", + "rustversion", + "wasm-bindgen-macro", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.71" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96492d0d3ffba25305a7dc88720d250b1401d7edca02cc3bcd50633b424673b8" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.121" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e68e6f4afd367a562002c05637acb8578ff2dea1943df76afb9e83d177c8578" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.121" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d95a9ec35c64b2a7cb35d3fead40c4238d0940c86d107136999567a4703259f2" +dependencies = [ + "bumpalo", + "proc-macro2", + "quote", + "syn 2.0.117", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.121" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4e0100b01e9f0d03189a92b96772a1fb998639d981193d7dbab487302513441" +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.1", + "hashbrown 0.15.5", + "indexmap 2.14.0", + "semver", +] + +[[package]] +name = "web-sys" +version = "0.3.98" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4b572dff8bcf38bad0fa19729c89bb5748b2b9b1d8be70cf90df697e3a8f32aa" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "web_atoms" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7cff6eef815df1834fd250e3a2ff436044d82a9f1bc1980ca1dbdf07effc538" +dependencies = [ + "phf 0.13.1", + "phf_codegen", + "string_cache", + "string_cache_codegen", +] + +[[package]] +name = "webkit2gtk" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1027150013530fb2eaf806408df88461ae4815a45c541c8975e61d6f2fc4793" +dependencies = [ + "bitflags 1.3.2", + "cairo-rs", + "gdk", + "gdk-sys", + "gio", + "gio-sys", + "glib", + "glib-sys", + "gobject-sys", + "gtk", + "gtk-sys", + "javascriptcore-rs", + "libc", + "once_cell", + "soup3", + "webkit2gtk-sys", +] + +[[package]] +name = "webkit2gtk-sys" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "916a5f65c2ef0dfe12fff695960a2ec3d4565359fdbb2e9943c974e06c734ea5" +dependencies = [ + "bitflags 1.3.2", + "cairo-sys-rs", + "gdk-sys", + "gio-sys", + "glib-sys", + "gobject-sys", + "gtk-sys", + "javascriptcore-rs-sys", + "libc", + "pkg-config", + "soup3-sys", + "system-deps", +] + +[[package]] +name = "webpki-root-certs" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f31141ce3fc3e300ae89b78c0dd67f9708061d1d2eda54b8209346fd6be9a92c" +dependencies = [ + "rustls-pki-types", +] + +[[package]] +name = "webview2-com" +version = "0.38.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7130243a7a5b33c54a444e54842e6a9e133de08b5ad7b5861cd8ed9a6a5bc96a" +dependencies = [ + "webview2-com-macros", + "webview2-com-sys", + "windows 0.61.3", + "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 0.61.3", + "windows-core 0.61.2", +] + +[[package]] +name = "whoami" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "998767ef88740d1f5b0682a9c53c24431453923962269c2db68ee43788c5a40d" +dependencies = [ + "libc", + "libredox", + "objc2-system-configuration", + "wasite", + "web-sys", +] + +[[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 0.2.0", + "windows-core 0.61.2", + "windows-future 0.2.1", + "windows-link 0.1.3", + "windows-numerics 0.2.0", +] + +[[package]] +name = "windows" +version = "0.62.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "527fadee13e0c05939a6a05d5bd6eec6cd2e3dbd648b9f8e447c6518133d8580" +dependencies = [ + "windows-collections 0.3.2", + "windows-core 0.62.2", + "windows-future 0.3.2", + "windows-numerics 0.3.1", +] + +[[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-collections" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23b2d95af1a8a14a3c7367e1ed4fc9c20e0a26e79551b1454d72583c97cc6610" +dependencies = [ + "windows-core 0.62.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 0.1.0", +] + +[[package]] +name = "windows-future" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e1d6f90251fe18a279739e78025bd6ddc52a7e22f921070ccdc67dde84c605cb" +dependencies = [ + "windows-core 0.62.2", + "windows-link 0.2.1", + "windows-threading 0.2.1", +] + +[[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-numerics" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e2e40844ac143cdb44aead537bbf727de9b044e107a0f1220392177d15b0f26" +dependencies = [ + "windows-core 0.62.2", + "windows-link 0.2.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.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.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-threading" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3949bd5b99cafdf1c7ca86b43ca564028dfe27d66958f2470940f73d86d75b37" +dependencies = [ + "windows-link 0.2.1", +] + +[[package]] +name = "windows-version" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e4060a1da109b9d0326b7262c8e12c84df67cc0dbc9e33cf49e01ccc2eb63631" +dependencies = [ + "windows-link 0.2.1", +] + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "597a5118570b68bc08d8d59125332c54f1ba9d9adeedeef5b99b02ba2b0698f8" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a9d8416fa8b42f5c947f8482c43e7d89e73a173cead56d044f6a56104a6d1b53" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e08e8864a60f06ef0d0ff4ba04124db8b0fb3be5776a5cd47641e942e58c4d43" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b9d782e804c2f632e395708e99a94275910eb9100b2114651e04744e9b125006" + +[[package]] +name = "windows_i686_gnu" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c61d927d8da41da96a81f029489353e68739737d3beca43145c8afec9a31a84f" + +[[package]] +name = "windows_i686_gnu" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b" + +[[package]] +name = "windows_i686_gnu" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "960e6da069d81e09becb0ca57a65220ddff016ff2d6af6a223cf372a506593a3" + +[[package]] +name = "windows_i686_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66" + +[[package]] +name = "windows_i686_gnullvm" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa7359d10048f68ab8b09fa71c3daccfb0e9b559aed648a8f95469c27057180c" + +[[package]] +name = "windows_i686_msvc" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44d840b6ec649f480a41c8d80f9c65108b92d89345dd94027bfe06ac444d1060" + +[[package]] +name = "windows_i686_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66" + +[[package]] +name = "windows_i686_msvc" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e7ac75179f18232fe9c285163565a57ef8d3c89254a30685b57d83a38d326c2" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8de912b8b8feb55c064867cf047dda097f92d51efad5b491dfb98f6bbb70cb36" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c3842cdd74a865a8066ab39c8a7a473c0778a3f29370b5fd6b4b9aa7df4a499" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26d41b46a36d453748aedef1486d5c7a85db22e56aff34643984ea85514e94a3" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ffa179e2d07eee8ad8f57493436566c7cc30ac536a3379fdf008f47f6bb7ae1" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9aec5da331524158c6d1a4ac0ab1541149c0b9505fde06423b02f5ef0106b9f0" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.53.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d6bbff5f0aada427a1e5a6da5f1f98158182f26556f345ac9e04d36d0ebed650" + +[[package]] +name = "winnow" +version = "0.5.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f593a95398737aeed53e489c785df13f3618e41dbcd6718c6addbf1395aa6876" +dependencies = [ + "memchr", +] + +[[package]] +name = "winnow" +version = "0.7.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df79d97927682d2fd8adb29682d1140b343be4ac0f08fd68b7765d9c059d3945" + +[[package]] +name = "winnow" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2ee1708bef14716a11bae175f579062d4554d95be2c6829f518df847b7b3fdd0" +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" +version = "0.57.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1ebf944e87a7c253233ad6766e082e3cd714b5d03812acc24c318f549614536e" + +[[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.1", + "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.55.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "186f9871daa55fd9c016578b810d149de58367113db7fb72b462d2323ce19514" +dependencies = [ + "base64 0.22.1", + "block2", + "cookie", + "crossbeam-channel", + "dirs", + "dom_query", + "dpi", + "dunce", + "gdkx11", + "gtk", + "http", + "javascriptcore-rs", + "jni 0.21.1", + "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 0.10.9", + "soup3", + "tao-macros", + "thiserror 2.0.18", + "url", + "webkit2gtk", + "webkit2gtk-sys", + "webview2-com", + "windows 0.61.3", + "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 = "xattr" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32e45ad4206f6d2479085147f02bc2ef834ac85886624a23575ae137c8aa8156" +dependencies = [ + "libc", + "rustix", +] + +[[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.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ec05a11813ea801ff6d75110ad09cd0824ddba17dfe17128ea0d5f68e6c5272" +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 = "zip" +version = "4.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "caa8cd6af31c3b31c6631b8f483848b91589021b28fffe50adada48d4f4d2ed1" +dependencies = [ + "arbitrary", + "crc32fast", + "indexmap 2.14.0", + "memchr", +] + +[[package]] +name = "zmij" +version = "1.0.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8848ee67ecc8aedbaf3e4122217aff892639231befc6a1b58d29fff4c2cabaa" diff --git a/docs/reference-repos/adainrivers/Cargo.toml b/docs/reference-repos/adainrivers/Cargo.toml new file mode 100644 index 0000000..2005601 --- /dev/null +++ b/docs/reference-repos/adainrivers/Cargo.toml @@ -0,0 +1,7 @@ +[workspace] +members = ["crates/dune-manager-core", "crates/dune-server-service", "app/src-tauri"] +resolver = "2" + +[workspace.dependencies] +serde = { version = "1", features = ["derive"] } +serde_json = "1" diff --git a/docs/reference-repos/adainrivers/LICENSE b/docs/reference-repos/adainrivers/LICENSE new file mode 100644 index 0000000..5cd827f --- /dev/null +++ b/docs/reference-repos/adainrivers/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2026 gaming.tools + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/docs/reference-repos/adainrivers/README.md b/docs/reference-repos/adainrivers/README.md new file mode 100644 index 0000000..fb8ad08 --- /dev/null +++ b/docs/reference-repos/adainrivers/README.md @@ -0,0 +1,59 @@ +# Dune Dedicated Server Manager + +A desktop manager for existing Dune Awakening dedicated servers. + +![Dashboard — BattleGroup status, lifecycle actions, management service, and tunnel controls](images/ss-1.png) + +The app manages already-provisioned Dune dedicated servers over SSH and +Kubernetes control commands. It does not install the game server, create VMs, +configure Hyper-V, provision Ubuntu, or manage external tools such as SteamCMD. + +## Features + +- Remote server profile management with SSH private-key authentication +- BattleGroup status, start, stop, restart, and update controls +- Component diagnostics, log viewing, and safe restart actions +- Secure Director, File Browser, PostgreSQL, and PgHero access through local SSH tunnels +- Bundled `dune-server-service` daemon for on-host scheduled maintenance (daily restarts with in-game warnings, automated backups, server update check + apply) — installed over SSH straight from the Management card +- Admin console for in-game actions: item grants, vehicle spawns, skill/journey/XP tags, player lookup with live pawn location, and a logged history of every published command +- Automated tasks tab with editable schedule settings (daily restart time, warning lead/frequency, update apply lead, IANA timezone) — saving auto-restarts the service so changes apply immediately +- Welcome Package automation: a per-player onboarding chain (item grants, water refill, welcome whisper) driven by Postgres player detection, tracked in the management service's SQLite ledger, and configurable from the Welcome Package tab with both a visual editor and a raw JSON mode + +![Admin tab — granting items to online players with a searchable Funcom item picker](images/ss-2.png) + +More management features coming soon. + +## Install + +Download the latest release for your operating system from GitHub Releases. + +- Windows: run the NSIS installer. +- Linux: use the AppImage or Debian package. +- macOS: use the DMG for your Mac architecture. + +After launching the app, add an existing server profile with its host, SSH user, +and private key path, then refresh it to detect BattleGroups and management +endpoints. + +## Managed Server Assumptions + +The target server must already be installed and reachable over SSH. The app +expects the Dune Kubernetes resources and vendor management scripts to exist on +the server before you add it. + +Required player-facing/server ports depend on your own server deployment. A +typical dedicated-server deployment uses: + +- UDP 7777-7810 for game servers +- TCP 31982 for RMQ + +If you found a bug or are having other issues, please create an issue here: +https://github.com/adainrivers/dune-dedicated-server-manager/issues + +## Building From Source + +See [Building From Source](docs/building-from-source.md). + +## License + +MIT License. See [LICENSE](LICENSE). diff --git a/docs/reference-repos/adainrivers/app/index.html b/docs/reference-repos/adainrivers/app/index.html new file mode 100644 index 0000000..a77f646 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/index.html @@ -0,0 +1,15 @@ + + + + + + Dune Dedicated Server Manager + + + + + +
+ + + diff --git a/docs/reference-repos/adainrivers/app/package-lock.json b/docs/reference-repos/adainrivers/app/package-lock.json new file mode 100644 index 0000000..1980ecf --- /dev/null +++ b/docs/reference-repos/adainrivers/app/package-lock.json @@ -0,0 +1,3897 @@ +{ + "name": "dune-dedicated-server-manager-app", + "version": "0.3.16", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "dune-dedicated-server-manager-app", + "version": "0.3.16", + "dependencies": { + "@radix-ui/react-icons": "^1.3.2", + "@radix-ui/themes": "^3.2.1", + "@tauri-apps/api": "^2.0.0", + "@tauri-apps/plugin-dialog": "^2.7.1", + "@tauri-apps/plugin-process": "^2.3.1", + "@tauri-apps/plugin-shell": "^2.3.5", + "@tauri-apps/plugin-updater": "^2.10.1", + "markdown-to-jsx": "^9.8.1", + "react": "^18.3.1", + "react-dom": "^18.3.1" + }, + "devDependencies": { + "@tauri-apps/cli": "^2.0.0", + "@types/react": "^18.3.12", + "@types/react-dom": "^18.3.1", + "@vitejs/plugin-react": "^4.3.3", + "typescript": "^5.6.3", + "vite": "^5.4.10" + } + }, + "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.3", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.3.tgz", + "integrity": "sha512-LIVqM46zQWZhj17qA8wb4nW/ixr2y1Nw+r1etiAWgRM6U1IqP+LNhL1yg440jYZR72jCWcWbLWzIosH+uP1fqg==", + "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", + "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.3", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.3.tgz", + "integrity": "sha512-b3ctpQwp+PROvU/cttc4OYl4MzfJUWy6FZg+PMXfzmt/+39iHVF0sDfqay8TQM3JA2EUOyKcFZt75jWriQijsA==", + "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.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", + "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", + "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", + "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", + "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", + "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", + "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", + "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", + "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", + "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", + "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", + "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", + "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", + "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", + "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", + "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", + "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", + "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", + "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", + "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", + "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", + "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", + "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", + "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@floating-ui/core": { + "version": "1.7.5", + "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.7.5.tgz", + "integrity": "sha512-1Ih4WTWyw0+lKyFMcBHGbb5U5FtuHJuujoyyr5zTaWS5EYMeT6Jb2AuDeftsCsEuchO+mM2ij5+q9crhydzLhQ==", + "license": "MIT", + "dependencies": { + "@floating-ui/utils": "^0.2.11" + } + }, + "node_modules/@floating-ui/dom": { + "version": "1.7.6", + "resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.7.6.tgz", + "integrity": "sha512-9gZSAI5XM36880PPMm//9dfiEngYoC6Am2izES1FF406YFsjvyBMmeJ2g4SAju3xWwtuynNRFL2s9hgxpLI5SQ==", + "license": "MIT", + "dependencies": { + "@floating-ui/core": "^1.7.5", + "@floating-ui/utils": "^0.2.11" + } + }, + "node_modules/@floating-ui/react-dom": { + "version": "2.1.8", + "resolved": "https://registry.npmjs.org/@floating-ui/react-dom/-/react-dom-2.1.8.tgz", + "integrity": "sha512-cC52bHwM/n/CxS87FH0yWdngEZrjdtLW/qVruo68qg+prK7ZQ4YGdut2GyDVpoGeAYe/h899rVeOVm6Oi40k2A==", + "license": "MIT", + "dependencies": { + "@floating-ui/dom": "^1.7.6" + }, + "peerDependencies": { + "react": ">=16.8.0", + "react-dom": ">=16.8.0" + } + }, + "node_modules/@floating-ui/utils": { + "version": "0.2.11", + "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.11.tgz", + "integrity": "sha512-RiB/yIh78pcIxl6lLMG0CgBXAZ2Y0eVHqMPYugu+9U0AeT6YBeiJpf7lbdJNIugFP5SIjwNRgo4DhR1Qxi26Gg==", + "license": "MIT" + }, + "node_modules/@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/@radix-ui/colors": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@radix-ui/colors/-/colors-3.0.0.tgz", + "integrity": "sha512-FUOsGBkHrYJwCSEtWRCIfQbZG7q1e6DgxCIOe1SUQzDe/7rXXeA47s8yCn6fuTNQAj1Zq4oTFi9Yjp3wzElcxg==", + "license": "MIT" + }, + "node_modules/@radix-ui/number": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/number/-/number-1.1.1.tgz", + "integrity": "sha512-MkKCwxlXTgz6CFoJx3pCwn07GKp36+aZyu/u2Ln2VrA5DcdyCZkASEDBTd8x5whTQQL5CiYf4prXKLcgQdv29g==", + "license": "MIT" + }, + "node_modules/@radix-ui/primitive": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@radix-ui/primitive/-/primitive-1.1.3.tgz", + "integrity": "sha512-JTF99U/6XIjCBo0wqkU5sK10glYe27MRRsfwoiq5zzOEZLHU3A3KCMa5X/azekYRCJ0HlwI0crAXS/5dEHTzDg==", + "license": "MIT" + }, + "node_modules/@radix-ui/react-accessible-icon": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-accessible-icon/-/react-accessible-icon-1.1.7.tgz", + "integrity": "sha512-XM+E4WXl0OqUJFovy6GjmxxFyx9opfCAIUku4dlKRd5YEPqt4kALOkQOp0Of6reHuUkJuiPBEc5k0o4z4lTC8A==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-visually-hidden": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-accordion": { + "version": "1.2.12", + "resolved": "https://registry.npmjs.org/@radix-ui/react-accordion/-/react-accordion-1.2.12.tgz", + "integrity": "sha512-T4nygeh9YE9dLRPhAHSeOZi7HBXo+0kYIPJXayZfvWOWA0+n3dESrZbjfDPUABkUNym6Hd+f2IR113To8D2GPA==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collapsible": "1.1.12", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-alert-dialog": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/@radix-ui/react-alert-dialog/-/react-alert-dialog-1.1.15.tgz", + "integrity": "sha512-oTVLkEw5GpdRe29BqJ0LSDFWI3qu0vR1M0mUkOQWDIUnY/QIkLpgDMWuKxP94c2NAC2LGcgVhG1ImF3jkZ5wXw==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-dialog": "1.1.15", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-slot": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-arrow": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-arrow/-/react-arrow-1.1.7.tgz", + "integrity": "sha512-F+M1tLhO+mlQaOWspE8Wstg+z6PwxwRd8oQ8IXceWz92kfAmalTRf0EjrouQeo7QssEPfCn05B4Ihs1K9WQ/7w==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-primitive": "2.1.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-aspect-ratio": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-aspect-ratio/-/react-aspect-ratio-1.1.7.tgz", + "integrity": "sha512-Yq6lvO9HQyPwev1onK1daHCHqXVLzPhSVjmsNjCa2Zcxy2f7uJD2itDtxknv6FzAKCwD1qQkeVDmX/cev13n/g==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-primitive": "2.1.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-avatar": { + "version": "1.1.10", + "resolved": "https://registry.npmjs.org/@radix-ui/react-avatar/-/react-avatar-1.1.10.tgz", + "integrity": "sha512-V8piFfWapM5OmNCXTzVQY+E1rDa53zY+MQ4Y7356v4fFz6vqCyUtIz2rUD44ZEdwg78/jKmMJHj07+C/Z/rcog==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-is-hydrated": "0.1.0", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-checkbox": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/@radix-ui/react-checkbox/-/react-checkbox-1.3.3.tgz", + "integrity": "sha512-wBbpv+NQftHDdG86Qc0pIyXk5IR3tM8Vd0nWLKDcX8nNn4nXFOFwsKuqw2okA/1D/mpaAkmuyndrPJTYDNZtFw==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-previous": "1.1.1", + "@radix-ui/react-use-size": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-collapsible": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/@radix-ui/react-collapsible/-/react-collapsible-1.1.12.tgz", + "integrity": "sha512-Uu+mSh4agx2ib1uIGPP4/CKNULyajb3p92LsVXmH2EHVMTfZWpll88XJ0j4W0z3f8NK1eYl1+Mf/szHPmcHzyA==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-collection": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-collection/-/react-collection-1.1.7.tgz", + "integrity": "sha512-Fh9rGN0MoI4ZFUNyfFVNU4y9LUz93u9/0K+yLgA2bwRojxM8JU1DyvvMBabnZPBgMWREAJvU2jjVzq+LrFUglw==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-slot": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-compose-refs": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-compose-refs/-/react-compose-refs-1.1.2.tgz", + "integrity": "sha512-z4eqJvfiNnFMHIIvXP3CY57y2WJs5g2v3X0zm9mEJkrkNv4rDxu+sg9Jh8EkXyeqBkB7SOcboo9dMVqhyrACIg==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-context": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-context/-/react-context-1.1.2.tgz", + "integrity": "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-context-menu": { + "version": "2.2.16", + "resolved": "https://registry.npmjs.org/@radix-ui/react-context-menu/-/react-context-menu-2.2.16.tgz", + "integrity": "sha512-O8morBEW+HsVG28gYDZPTrT9UUovQUlJue5YO836tiTJhuIWBm/zQHc7j388sHWtdH/xUZurK9olD2+pcqx5ww==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-menu": "2.1.16", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-dialog": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/@radix-ui/react-dialog/-/react-dialog-1.1.15.tgz", + "integrity": "sha512-TCglVRtzlffRNxRMEyR36DGBLJpeusFcgMVD9PZEzAKnUs1lKCgX5u9BmC2Yg+LL9MgZDugFFs1Vl+Jp4t/PGw==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-focus-guards": "1.1.3", + "@radix-ui/react-focus-scope": "1.1.7", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-slot": "1.2.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "aria-hidden": "^1.2.4", + "react-remove-scroll": "^2.6.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-direction": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-direction/-/react-direction-1.1.1.tgz", + "integrity": "sha512-1UEWRX6jnOA2y4H5WczZ44gOOjTEmlqv1uNW4GAJEO5+bauCBhv8snY65Iw5/VOS/ghKN9gr2KjnLKxrsvoMVw==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-dismissable-layer": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/@radix-ui/react-dismissable-layer/-/react-dismissable-layer-1.1.11.tgz", + "integrity": "sha512-Nqcp+t5cTB8BinFkZgXiMJniQH0PsUt2k51FUhbdfeKvc4ACcG2uQniY/8+h1Yv6Kza4Q7lD7PQV0z0oicE0Mg==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-escape-keydown": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-dropdown-menu": { + "version": "2.1.16", + "resolved": "https://registry.npmjs.org/@radix-ui/react-dropdown-menu/-/react-dropdown-menu-2.1.16.tgz", + "integrity": "sha512-1PLGQEynI/3OX/ftV54COn+3Sud/Mn8vALg2rWnBLnRaGtJDduNW/22XjlGgPdpcIbiQxjKtb7BkcjP00nqfJw==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-menu": "2.1.16", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-focus-guards": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@radix-ui/react-focus-guards/-/react-focus-guards-1.1.3.tgz", + "integrity": "sha512-0rFg/Rj2Q62NCm62jZw0QX7a3sz6QCQU0LpZdNrJX8byRGaGVTqbrW9jAoIAHyMQqsNpeZ81YgSizOt5WXq0Pw==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-focus-scope": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-focus-scope/-/react-focus-scope-1.1.7.tgz", + "integrity": "sha512-t2ODlkXBQyn7jkl6TNaw/MtVEVvIGelJDCG41Okq/KwUsJBwQ4XVZsHAVUkK4mBv3ewiAS3PGuUWuY2BoK4ZUw==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-form": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/@radix-ui/react-form/-/react-form-0.1.8.tgz", + "integrity": "sha512-QM70k4Zwjttifr5a4sZFts9fn8FzHYvQ5PiB19O2HsYibaHSVt9fH9rzB0XZo/YcM+b7t/p7lYCT/F5eOeF5yQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-label": "2.1.7", + "@radix-ui/react-primitive": "2.1.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-hover-card": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/@radix-ui/react-hover-card/-/react-hover-card-1.1.15.tgz", + "integrity": "sha512-qgTkjNT1CfKMoP0rcasmlH2r1DAiYicWsDsufxl940sT2wHNEWWv6FMWIQXWhVdmC1d/HYfbhQx60KYyAtKxjg==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-popper": "1.2.8", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-icons": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-icons/-/react-icons-1.3.2.tgz", + "integrity": "sha512-fyQIhGDhzfc9pK2kH6Pl9c4BDJGfMkPqkyIgYDthyNYoNg3wVhoJMMh19WS4Up/1KMPFVpNsT2q3WmXn2N1m6g==", + "license": "MIT", + "peerDependencies": { + "react": "^16.x || ^17.x || ^18.x || ^19.0.0 || ^19.0.0-rc" + } + }, + "node_modules/@radix-ui/react-id": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-id/-/react-id-1.1.1.tgz", + "integrity": "sha512-kGkGegYIdQsOb4XjsfM97rXsiHaBwco+hFI66oO4s9LU+PLAC5oJ7khdOVFxkhsmlbpUqDAvXw11CluXP+jkHg==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-label": { + "version": "2.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-label/-/react-label-2.1.7.tgz", + "integrity": "sha512-YT1GqPSL8kJn20djelMX7/cTRp/Y9w5IZHvfxQTVHrOqa2yMl7i/UfMqKRU5V7mEyKTrUVgJXhNQPVCG8PBLoQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-primitive": "2.1.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-menu": { + "version": "2.1.16", + "resolved": "https://registry.npmjs.org/@radix-ui/react-menu/-/react-menu-2.1.16.tgz", + "integrity": "sha512-72F2T+PLlphrqLcAotYPp0uJMr5SjP5SL01wfEspJbru5Zs5vQaSHb4VB3ZMJPimgHHCHG7gMOeOB9H3Hdmtxg==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-focus-guards": "1.1.3", + "@radix-ui/react-focus-scope": "1.1.7", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-popper": "1.2.8", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-slot": "1.2.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "aria-hidden": "^1.2.4", + "react-remove-scroll": "^2.6.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-menubar": { + "version": "1.1.16", + "resolved": "https://registry.npmjs.org/@radix-ui/react-menubar/-/react-menubar-1.1.16.tgz", + "integrity": "sha512-EB1FktTz5xRRi2Er974AUQZWg2yVBb1yjip38/lgwtCVRd3a+maUoGHN/xs9Yv8SY8QwbSEb+YrxGadVWbEutA==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-menu": "2.1.16", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-navigation-menu": { + "version": "1.2.14", + "resolved": "https://registry.npmjs.org/@radix-ui/react-navigation-menu/-/react-navigation-menu-1.2.14.tgz", + "integrity": "sha512-YB9mTFQvCOAQMHU+C/jVl96WmuWeltyUEpRJJky51huhds5W2FQr1J8D/16sQlf0ozxkPK8uF3niQMdUwZPv5w==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-layout-effect": "1.1.1", + "@radix-ui/react-use-previous": "1.1.1", + "@radix-ui/react-visually-hidden": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-one-time-password-field": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/@radix-ui/react-one-time-password-field/-/react-one-time-password-field-0.1.8.tgz", + "integrity": "sha512-ycS4rbwURavDPVjCb5iS3aG4lURFDILi6sKI/WITUMZ13gMmn/xGjpLoqBAalhJaDk8I3UbCM5GzKHrnzwHbvg==", + "license": "MIT", + "dependencies": { + "@radix-ui/number": "1.1.1", + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-effect-event": "0.0.2", + "@radix-ui/react-use-is-hydrated": "0.1.0", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-password-toggle-field": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@radix-ui/react-password-toggle-field/-/react-password-toggle-field-0.1.3.tgz", + "integrity": "sha512-/UuCrDBWravcaMix4TdT+qlNdVwOM1Nck9kWx/vafXsdfj1ChfhOdfi3cy9SGBpWgTXwYCuboT/oYpJy3clqfw==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-effect-event": "0.0.2", + "@radix-ui/react-use-is-hydrated": "0.1.0" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-popover": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/@radix-ui/react-popover/-/react-popover-1.1.15.tgz", + "integrity": "sha512-kr0X2+6Yy/vJzLYJUPCZEc8SfQcf+1COFoAqauJm74umQhta9M7lNJHP7QQS3vkvcGLQUbWpMzwrXYwrYztHKA==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-focus-guards": "1.1.3", + "@radix-ui/react-focus-scope": "1.1.7", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-popper": "1.2.8", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-slot": "1.2.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "aria-hidden": "^1.2.4", + "react-remove-scroll": "^2.6.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-popper": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@radix-ui/react-popper/-/react-popper-1.2.8.tgz", + "integrity": "sha512-0NJQ4LFFUuWkE7Oxf0htBKS6zLkkjBH+hM1uk7Ng705ReR8m/uelduy1DBo0PyBXPKVnBA6YBlU94MBGXrSBCw==", + "license": "MIT", + "dependencies": { + "@floating-ui/react-dom": "^2.0.0", + "@radix-ui/react-arrow": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-layout-effect": "1.1.1", + "@radix-ui/react-use-rect": "1.1.1", + "@radix-ui/react-use-size": "1.1.1", + "@radix-ui/rect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-portal": { + "version": "1.1.9", + "resolved": "https://registry.npmjs.org/@radix-ui/react-portal/-/react-portal-1.1.9.tgz", + "integrity": "sha512-bpIxvq03if6UNwXZ+HTK71JLh4APvnXntDc6XOX8UVq4XQOVl7lwok0AvIl+b8zgCw3fSaVTZMpAPPagXbKmHQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-presence": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/@radix-ui/react-presence/-/react-presence-1.1.5.tgz", + "integrity": "sha512-/jfEwNDdQVBCNvjkGit4h6pMOzq8bHkopq458dPt2lMjx+eBQUohZNG9A7DtO/O5ukSbxuaNGXMjHicgwy6rQQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-primitive": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/@radix-ui/react-primitive/-/react-primitive-2.1.3.tgz", + "integrity": "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-slot": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-progress": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-progress/-/react-progress-1.1.7.tgz", + "integrity": "sha512-vPdg/tF6YC/ynuBIJlk1mm7Le0VgW6ub6J2UWnTQ7/D23KXcPI1qy+0vBkgKgd38RCMJavBXpB83HPNFMTb0Fg==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-primitive": "2.1.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-radio-group": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/@radix-ui/react-radio-group/-/react-radio-group-1.3.8.tgz", + "integrity": "sha512-VBKYIYImA5zsxACdisNQ3BjCBfmbGH3kQlnFVqlWU4tXwjy7cGX8ta80BcrO+WJXIn5iBylEH3K6ZTlee//lgQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-previous": "1.1.1", + "@radix-ui/react-use-size": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-roving-focus": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/@radix-ui/react-roving-focus/-/react-roving-focus-1.1.11.tgz", + "integrity": "sha512-7A6S9jSgm/S+7MdtNDSb+IU859vQqJ/QAtcYQcfFC6W8RS4IxIZDldLR0xqCFZ6DCyrQLjLPsxtTNch5jVA4lA==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-scroll-area": { + "version": "1.2.10", + "resolved": "https://registry.npmjs.org/@radix-ui/react-scroll-area/-/react-scroll-area-1.2.10.tgz", + "integrity": "sha512-tAXIa1g3sM5CGpVT0uIbUx/U3Gs5N8T52IICuCtObaos1S8fzsrPXG5WObkQN3S6NVl6wKgPhAIiBGbWnvc97A==", + "license": "MIT", + "dependencies": { + "@radix-ui/number": "1.1.1", + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-select": { + "version": "2.2.6", + "resolved": "https://registry.npmjs.org/@radix-ui/react-select/-/react-select-2.2.6.tgz", + "integrity": "sha512-I30RydO+bnn2PQztvo25tswPH+wFBjehVGtmagkU78yMdwTwVf12wnAOF+AeP8S2N8xD+5UPbGhkUfPyvT+mwQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/number": "1.1.1", + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-focus-guards": "1.1.3", + "@radix-ui/react-focus-scope": "1.1.7", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-popper": "1.2.8", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-slot": "1.2.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-layout-effect": "1.1.1", + "@radix-ui/react-use-previous": "1.1.1", + "@radix-ui/react-visually-hidden": "1.2.3", + "aria-hidden": "^1.2.4", + "react-remove-scroll": "^2.6.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-separator": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/@radix-ui/react-separator/-/react-separator-1.1.7.tgz", + "integrity": "sha512-0HEb8R9E8A+jZjvmFCy/J4xhbXy3TV+9XSnGJ3KvTtjlIUy/YQ/p6UYZvi7YbeoeXdyU9+Y3scizK6hkY37baA==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-primitive": "2.1.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-slider": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/@radix-ui/react-slider/-/react-slider-1.3.6.tgz", + "integrity": "sha512-JPYb1GuM1bxfjMRlNLE+BcmBC8onfCi60Blk7OBqi2MLTFdS+8401U4uFjnwkOr49BLmXxLC6JHkvAsx5OJvHw==", + "license": "MIT", + "dependencies": { + "@radix-ui/number": "1.1.1", + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-layout-effect": "1.1.1", + "@radix-ui/react-use-previous": "1.1.1", + "@radix-ui/react-use-size": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-slot": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.3.tgz", + "integrity": "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-compose-refs": "1.1.2" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-switch": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@radix-ui/react-switch/-/react-switch-1.2.6.tgz", + "integrity": "sha512-bByzr1+ep1zk4VubeEVViV592vu2lHE2BZY5OnzehZqOOgogN80+mNtCqPkhn2gklJqOpxWgPoYTSnhBCqpOXQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-previous": "1.1.1", + "@radix-ui/react-use-size": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-tabs": { + "version": "1.1.13", + "resolved": "https://registry.npmjs.org/@radix-ui/react-tabs/-/react-tabs-1.1.13.tgz", + "integrity": "sha512-7xdcatg7/U+7+Udyoj2zodtI9H/IIopqo+YOIcZOq1nJwXWBZ9p8xiu5llXlekDbZkca79a/fozEYQXIA4sW6A==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-toast": { + "version": "1.2.15", + "resolved": "https://registry.npmjs.org/@radix-ui/react-toast/-/react-toast-1.2.15.tgz", + "integrity": "sha512-3OSz3TacUWy4WtOXV38DggwxoqJK4+eDkNMl5Z/MJZaoUPaP4/9lf81xXMe1I2ReTAptverZUpbPY4wWwWyL5g==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-layout-effect": "1.1.1", + "@radix-ui/react-visually-hidden": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-toggle": { + "version": "1.1.10", + "resolved": "https://registry.npmjs.org/@radix-ui/react-toggle/-/react-toggle-1.1.10.tgz", + "integrity": "sha512-lS1odchhFTeZv3xwHH31YPObmJn8gOg7Lq12inrr0+BH/l3Tsq32VfjqH1oh80ARM3mlkfMic15n0kg4sD1poQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-toggle-group": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/@radix-ui/react-toggle-group/-/react-toggle-group-1.1.11.tgz", + "integrity": "sha512-5umnS0T8JQzQT6HbPyO7Hh9dgd82NmS36DQr+X/YJ9ctFNCiiQd6IJAYYZ33LUwm8M+taCz5t2ui29fHZc4Y6Q==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-toggle": "1.1.10", + "@radix-ui/react-use-controllable-state": "1.2.2" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-toolbar": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/@radix-ui/react-toolbar/-/react-toolbar-1.1.11.tgz", + "integrity": "sha512-4ol06/1bLoFu1nwUqzdD4Y5RZ9oDdKeiHIsntug54Hcr1pgaHiPqHFEaXI1IFP/EsOfROQZ8Mig9VTIRza6Tjg==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-separator": "1.1.7", + "@radix-ui/react-toggle-group": "1.1.11" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-tooltip": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@radix-ui/react-tooltip/-/react-tooltip-1.2.8.tgz", + "integrity": "sha512-tY7sVt1yL9ozIxvmbtN5qtmH2krXcBCfjEiCgKGLqunJHvgvZG2Pcl2oQ3kbcZARb1BGEHdkLzcYGO8ynVlieg==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-id": "1.1.1", + "@radix-ui/react-popper": "1.2.8", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-slot": "1.2.3", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-visually-hidden": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-callback-ref": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-callback-ref/-/react-use-callback-ref-1.1.1.tgz", + "integrity": "sha512-FkBMwD+qbGQeMu1cOHnuGB6x4yzPjho8ap5WtbEJ26umhgqVXbhekKUQO+hZEL1vU92a3wHwdp0HAcqAUF5iDg==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-controllable-state": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-controllable-state/-/react-use-controllable-state-1.2.2.tgz", + "integrity": "sha512-BjasUjixPFdS+NKkypcyyN5Pmg83Olst0+c6vGov0diwTEo6mgdqVR6hxcEgFuh4QrAs7Rc+9KuGJ9TVCj0Zzg==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-effect-event": "0.0.2", + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-effect-event": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-effect-event/-/react-use-effect-event-0.0.2.tgz", + "integrity": "sha512-Qp8WbZOBe+blgpuUT+lw2xheLP8q0oatc9UpmiemEICxGvFLYmHm9QowVZGHtJlGbS6A6yJ3iViad/2cVjnOiA==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-escape-keydown": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-escape-keydown/-/react-use-escape-keydown-1.1.1.tgz", + "integrity": "sha512-Il0+boE7w/XebUHyBjroE+DbByORGR9KKmITzbR7MyQ4akpORYP/ZmbhAr0DG7RmmBqoOnZdy2QlvajJ2QA59g==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-callback-ref": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-is-hydrated": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-is-hydrated/-/react-use-is-hydrated-0.1.0.tgz", + "integrity": "sha512-U+UORVEq+cTnRIaostJv9AGdV3G6Y+zbVd+12e18jQ5A3c0xL03IhnHuiU4UV69wolOQp5GfR58NW/EgdQhwOA==", + "license": "MIT", + "dependencies": { + "use-sync-external-store": "^1.5.0" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-layout-effect": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-layout-effect/-/react-use-layout-effect-1.1.1.tgz", + "integrity": "sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-previous": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-previous/-/react-use-previous-1.1.1.tgz", + "integrity": "sha512-2dHfToCj/pzca2Ck724OZ5L0EVrr3eHRNsG/b3xQJLA2hZpVCS99bLAX+hm1IHXDEnzU6by5z/5MIY794/a8NQ==", + "license": "MIT", + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-rect": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-rect/-/react-use-rect-1.1.1.tgz", + "integrity": "sha512-QTYuDesS0VtuHNNvMh+CjlKJ4LJickCMUAqjlE3+j8w+RlRpwyX3apEQKGFzbZGdo7XNG1tXa+bQqIE7HIXT2w==", + "license": "MIT", + "dependencies": { + "@radix-ui/rect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-use-size": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/react-use-size/-/react-use-size-1.1.1.tgz", + "integrity": "sha512-ewrXRDTAqAXlkl6t/fkXWNAhFX9I+CkKlw6zjEwk86RSPKwZr3xpBRso655aqYafwtnbpHLj6toFzmd6xdVptQ==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-use-layout-effect": "1.1.1" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/@radix-ui/react-visually-hidden": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/@radix-ui/react-visually-hidden/-/react-visually-hidden-1.2.3.tgz", + "integrity": "sha512-pzJq12tEaaIhqjbzpCuv/OypJY/BPavOofm+dbab+MHLajy277+1lLm6JFcGgF5eskJ6mquGirhXY2GD/8u8Ug==", + "license": "MIT", + "dependencies": { + "@radix-ui/react-primitive": "2.1.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "node_modules/@radix-ui/rect": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@radix-ui/rect/-/rect-1.1.1.tgz", + "integrity": "sha512-HPwpGIzkl28mWyZqG52jiqDJ12waP11Pa1lGoiyUkIEuMLBP0oeK/C89esbXrxsky5we7dfd8U58nm0SgAWpVw==", + "license": "MIT" + }, + "node_modules/@radix-ui/themes": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/@radix-ui/themes/-/themes-3.3.0.tgz", + "integrity": "sha512-I0/h2CRNTpYNB7Mi3xFIvSsQq5a108d7kK8dTO5zp5b9HR5QJXKag6B8tjpz2ITkVYkFdkGk45doNkSr7OxwNw==", + "license": "MIT", + "dependencies": { + "@radix-ui/colors": "^3.0.0", + "classnames": "^2.3.2", + "radix-ui": "^1.1.3", + "react-remove-scroll-bar": "^2.3.8" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "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.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.60.3.tgz", + "integrity": "sha512-x35CNW/ANXG3hE/EZpRU8MXX1JDN86hBb2wMGAtltkz7pc6cxgjpy1OMMfDosOQ+2hWqIkag/fGok1Yady9nGw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.60.3.tgz", + "integrity": "sha512-xw3xtkDApIOGayehp2+Rz4zimfkaX65r4t47iy+ymQB2G4iJCBBfj0ogVg5jpvjpn8UWn/+q9tprxleYeNp3Hw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.60.3.tgz", + "integrity": "sha512-vo6Y5Qfpx7/5EaamIwi0WqW2+zfiusVihKatLvtN1VFVy3D13uERk/6gZLU1UiHRL6fDXqj/ELIeVRGnvcTE1g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.60.3.tgz", + "integrity": "sha512-D+0QGcZhBzTN82weOnsSlY7V7+RMmPuF1CkbxyMAGE8+ZHeUjyb76ZiWmBlCu//AQQONvxcqRbwZTajZKqjuOw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.60.3.tgz", + "integrity": "sha512-6HnvHCT7fDyj6R0Ph7A6x8dQS/S38MClRWeDLqc0MdfWkxjiu1HSDYrdPhqSILzjTIC/pnXbbJbo+ft+gy/9hQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.60.3.tgz", + "integrity": "sha512-KHLgC3WKlUYW3ShFKnnosZDOJ0xjg9zp7au3sIm2bs/tGBeC2ipmvRh/N7JKi0t9Ue20C0dpEshi8WUubg+cnA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.60.3.tgz", + "integrity": "sha512-DV6fJoxEYWJOvaZIsok7KrYl0tPvga5OZ2yvKHNNYyk/2roMLqQAbGhr78EQ5YhHpnhLKJD3S1WFusAkmUuV5g==", + "cpu": [ + "arm" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.60.3.tgz", + "integrity": "sha512-mQKoJAzvuOs6F+TZybQO4GOTSMUu7v0WdxEk24krQ/uUxXoPTtHjuaUuPmFhtBcM4K0ons8nrE3JyhTuCFtT/w==", + "cpu": [ + "arm" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.60.3.tgz", + "integrity": "sha512-Whjj2qoiJ6+OOJMGptTYazaJvjOJm+iKHpXQM1P3LzGjt7Ff++Tp7nH4N8J/BUA7R9IHfDyx4DJIflifwnbmIA==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.60.3.tgz", + "integrity": "sha512-4YTNHKqGng5+yiZt3mg77nmyuCfmNfX4fPmyUapBcIk+BdwSwmCWGXOUxhXbBEkFHtoN5boLj/5NON+u5QC9tg==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-gnu": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.60.3.tgz", + "integrity": "sha512-SU3kNlhkpI4UqlUc2VXPGK9o886ZsSeGfMAX2ba2b8DKmMXq4AL7KUrkSWVbb7koVqx41Yczx6dx5PNargIrEA==", + "cpu": [ + "loong64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-musl": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.60.3.tgz", + "integrity": "sha512-6lDLl5h4TXpB1mTf2rQWnAk/LcXrx9vBfu/DT5TIPhvMhRWaZ5MxkIc8u4lJAmBo6klTe1ywXIUHFjylW505sg==", + "cpu": [ + "loong64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-gnu": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.60.3.tgz", + "integrity": "sha512-BMo8bOw8evlup/8G+cj5xWtPyp93xPdyoSN16Zy90Q2QZ0ZYRhCt6ZJSwbrRzG9HApFabjwj2p25TUPDWrhzqQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-musl": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.60.3.tgz", + "integrity": "sha512-E0L8X1dZN1/Rph+5VPF6Xj2G7JJvMACVXtamTJIDrVI44Y3K+G8gQaMEAavbqCGTa16InptiVrX6eM6pmJ+7qA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.60.3.tgz", + "integrity": "sha512-oZJ/WHaVfHUiRAtmTAeo3DcevNsVvH8mbvodjZy7D5QKvCefO371SiKRpxoDcCxB3PTRTLayWBkvmDQKTcX/sw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.60.3.tgz", + "integrity": "sha512-Dhbyh7j9FybM3YaTgaHmVALwA8AkUwTPccyCQ79TG9AJUsMQqgN1DDEZNr4+QUfwiWvLDumW5vdwzoeUF+TNxQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.60.3.tgz", + "integrity": "sha512-cJd1X5XhHHlltkaypz1UcWLA8AcoIi1aWhsvaWDskD1oz2eKCypnqvTQ8ykMNI0RSmm7NkTdSqSSD7zM0xa6Ig==", + "cpu": [ + "s390x" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.60.3.tgz", + "integrity": "sha512-DAZDBHQfG2oQuhY7mc6I3/qB4LU2fQCjRvxbDwd/Jdvb9fypP4IJ4qmtu6lNjes6B531AI8cg1aKC2di97bUxA==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.60.3.tgz", + "integrity": "sha512-cRxsE8c13mZOh3vP+wLDxpQBRrOHDIGOWyDL93Sy0Ga8y515fBcC2pjUfFwUe5T7tqvTvWbCpg1URM/AXdWIXA==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-openbsd-x64": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.60.3.tgz", + "integrity": "sha512-QaWcIgRxqEdQdhJqW4DJctsH6HCmo5vHxY0krHSX4jMtOqfzC+dqDGuHM87bu4H8JBeibWx7jFz+h6/4C8wA5Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ] + }, + "node_modules/@rollup/rollup-openharmony-arm64": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.60.3.tgz", + "integrity": "sha512-AaXwSvUi3QIPtroAUw1t5yHGIyqKEXwH54WUocFolZhpGDruJcs8c+xPNDRn4XiQsS7MEwnYsHW2l0MBLDMkWg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.60.3.tgz", + "integrity": "sha512-65LAKM/bAWDqKNEelHlcHvm2V+Vfb8C6INFxQXRHCvaVN1rJfwr4NvdP4FyzUaLqWfaCGaadf6UbTm8xJeYfEg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.60.3.tgz", + "integrity": "sha512-EEM2gyhBF5MFnI6vMKdX1LAosE627RGBzIoGMdLloPZkXrUN0Ckqgr2Qi8+J3zip/8NVVro3/FjB+tjhZUgUHA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-gnu": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.60.3.tgz", + "integrity": "sha512-E5Eb5H/DpxaoXH++Qkv28RcUJboMopmdDUALBczvHMf7hNIxaDZqwY5lK12UK1BHacSmvupoEWGu+n993Z0y1A==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.60.3.tgz", + "integrity": "sha512-hPt/bgL5cE+Qp+/TPHBqptcAgPzgj46mPcg/16zNUmbQk0j+mOEQV/+Lqu8QRtDV3Ek95Q6FeFITpuhl6OTsAA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@tauri-apps/api": { + "version": "2.11.0", + "resolved": "https://registry.npmjs.org/@tauri-apps/api/-/api-2.11.0.tgz", + "integrity": "sha512-7CinYODhky9lmO23xHnUFv0Xt43fbtWMyxZcLcRBlFkcgXKuEirBvHpmtJ89YMhyeGcq20Wuc47Fa4XjyniywA==", + "license": "Apache-2.0 OR MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/tauri" + } + }, + "node_modules/@tauri-apps/cli": { + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli/-/cli-2.11.1.tgz", + "integrity": "sha512-rpEbaJ/HzNb6fwsquwoAbq29/Vt4gADhS423A8fdkwL4edJ0wZmoB8ar7O6JPDL834MUKOCm/rrJ7c9oAaEaYQ==", + "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.11.1", + "@tauri-apps/cli-darwin-x64": "2.11.1", + "@tauri-apps/cli-linux-arm-gnueabihf": "2.11.1", + "@tauri-apps/cli-linux-arm64-gnu": "2.11.1", + "@tauri-apps/cli-linux-arm64-musl": "2.11.1", + "@tauri-apps/cli-linux-riscv64-gnu": "2.11.1", + "@tauri-apps/cli-linux-x64-gnu": "2.11.1", + "@tauri-apps/cli-linux-x64-musl": "2.11.1", + "@tauri-apps/cli-win32-arm64-msvc": "2.11.1", + "@tauri-apps/cli-win32-ia32-msvc": "2.11.1", + "@tauri-apps/cli-win32-x64-msvc": "2.11.1" + } + }, + "node_modules/@tauri-apps/cli-darwin-arm64": { + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-darwin-arm64/-/cli-darwin-arm64-2.11.1.tgz", + "integrity": "sha512-6eEKMBXsQPCuM1EmvrjT2+aBuxWQuFdKdW8pzNuNQtpq45nEEpBlD5gr8pUeAyOU1DQKlkFaEc/MPBxb/Pfjtg==", + "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.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-darwin-x64/-/cli-darwin-x64-2.11.1.tgz", + "integrity": "sha512-LQUO7exfRWjWALNhetph5guWpMeHphRpokOLk0OIbTTExaNwJNFu3I4vb+CCM/4G/QGoZe/5XikZOJdNEFP1ig==", + "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.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-arm-gnueabihf/-/cli-linux-arm-gnueabihf-2.11.1.tgz", + "integrity": "sha512-5i/awiBCRRhOUG8yjn0fMHXIWD5Ez8eEk5LtvOxyQrKuJkRaZDvnbIjZbE183blAwkoA4xN3aO/prJiqscl02Q==", + "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.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-arm64-gnu/-/cli-linux-arm64-gnu-2.11.1.tgz", + "integrity": "sha512-9LrwDw3S9Fygtw/Q6WDhOP+3svJRGAsejeE+GKrc0eO1ThMVhwi2LL6hw4dlKw93IfS7VY1G19sWGxJ/NcU4nA==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-arm64-musl": { + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-arm64-musl/-/cli-linux-arm64-musl-2.11.1.tgz", + "integrity": "sha512-mNA5dbbqPqDUdTIwdUYYuhO2GvIe9UnB2r0VU2njxBOS3Opbx4gKNC5yP0Iu4rYmEmqdlwry9VzGZQ3wq9dyFg==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-riscv64-gnu": { + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-riscv64-gnu/-/cli-linux-riscv64-gnu-2.11.1.tgz", + "integrity": "sha512-fZj3Gwq+6fUs305T5WQiD5iSGJw+j/4w/HGmk4sHDAcy+rp9zU5eaxB7nOyz5/I/nkNAuKPqfp6uIbiUBXkBCw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-x64-gnu": { + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-x64-gnu/-/cli-linux-x64-gnu-2.11.1.tgz", + "integrity": "sha512-XFxGxOvHM7jjeD6ozCKdGfhzJ7lERYDGZl1/Kb4fsvchaJsfLJ981TlyTG8Qy/gFq+f5GitH3bfrX9JAkjPEyw==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-linux-x64-musl": { + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-linux-x64-musl/-/cli-linux-x64-musl-2.11.1.tgz", + "integrity": "sha512-d5C2/Zm+68v7R9wTuTCjRQEVrWjcdMkJBZ1+rXse+QdMMlTB9+u9PDNDLw9PQflWxYLaYZ7tjxxL9Nb9II6PbA==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "Apache-2.0 OR MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tauri-apps/cli-win32-arm64-msvc": { + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-win32-arm64-msvc/-/cli-win32-arm64-msvc-2.11.1.tgz", + "integrity": "sha512-YdeVWFAR1pTXzUU6NLstPq4G6OLxuDrXCXEBdmBH+5EZIDXUx0D2kJlz3+YjpazkKvAzYpgziTsyRagls0OfRQ==", + "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.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-win32-ia32-msvc/-/cli-win32-ia32-msvc-2.11.1.tgz", + "integrity": "sha512-VBGkuH0eB9K9LLSMv361Gzr5Ou72sCS4+ztpmkWEQ+wd/amhcYOsf3X6qn1RJZDzIhiOYHJEOysZUC3baD01rA==", + "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.11.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/cli-win32-x64-msvc/-/cli-win32-x64-msvc-2.11.1.tgz", + "integrity": "sha512-b3ORhIAKgp9ZYY+zBt7b7r0kLU2kjvyGF0+MS2SBym3emsweGPybEqocJcmtMuxyBhkOKHP4CiuEJEDuAlTx6A==", + "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.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/plugin-dialog/-/plugin-dialog-2.7.1.tgz", + "integrity": "sha512-OK1UBXYt+ojcmxMktzzuyonYIFta8CmAASpX+CA+DTGK24KlHjhYI6x2iOJ/TjZF4N7/ACK1oFmEOjIY9IhzOQ==", + "license": "MIT OR Apache-2.0", + "dependencies": { + "@tauri-apps/api": "^2.11.0" + } + }, + "node_modules/@tauri-apps/plugin-process": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/plugin-process/-/plugin-process-2.3.1.tgz", + "integrity": "sha512-nCa4fGVaDL/B9ai03VyPOjfAHRHSBz5v6F/ObsB73r/dA3MHHhZtldaDMIc0V/pnUw9ehzr2iEG+XkSEyC0JJA==", + "license": "MIT OR Apache-2.0", + "dependencies": { + "@tauri-apps/api": "^2.8.0" + } + }, + "node_modules/@tauri-apps/plugin-shell": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@tauri-apps/plugin-shell/-/plugin-shell-2.3.5.tgz", + "integrity": "sha512-jewtULhiQ7lI7+owCKAjc8tYLJr92U16bPOeAa472LHJdgaibLP83NcfAF2e+wkEcA53FxKQAZ7byDzs2eeizg==", + "license": "MIT OR Apache-2.0", + "dependencies": { + "@tauri-apps/api": "^2.10.1" + } + }, + "node_modules/@tauri-apps/plugin-updater": { + "version": "2.10.1", + "resolved": "https://registry.npmjs.org/@tauri-apps/plugin-updater/-/plugin-updater-2.10.1.tgz", + "integrity": "sha512-NFYMg+tWOZPJdzE/PpFj2qfqwAWwNS3kXrb1tm1gnBJ9mYzZ4WDRrwy8udzWoAnfGCHLuePNLY1WVCNHnh3eRA==", + "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==", + "devOptional": 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==", + "devOptional": true, + "license": "MIT", + "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==", + "devOptional": 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/aria-hidden": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/aria-hidden/-/aria-hidden-1.2.6.tgz", + "integrity": "sha512-ik3ZgC9dY/lYVVM++OISsaYDeg1tb0VtP5uL3ouh1koGOaUMDPpbFIei4JkFimWUFPn90sbMNMXQAIVOlnYKJA==", + "license": "MIT", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/baseline-browser-mapping": { + "version": "2.10.29", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.29.tgz", + "integrity": "sha512-Asa2krT+XTPZINCS+2QcyS8WTkObE77RwkydwF7h6DmnKqbvlalz93m/dnphUyCa6SWSP51VgtEUf2FN+gelFQ==", + "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", + "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.30001792", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001792.tgz", + "integrity": "sha512-hVLMUZFgR4JJ6ACt1uEESvQN1/dBVqPAKY0hgrV70eN3391K6juAfTjKZLKvOMsx8PxA7gsY1/tLMMTcfFLLpw==", + "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/classnames": { + "version": "2.5.1", + "resolved": "https://registry.npmjs.org/classnames/-/classnames-2.5.1.tgz", + "integrity": "sha512-saHYOzhIQs6wy2sVxTM6bUDsQO4F50V9RQ22qBpEdCW+I+/Wmke2HOl6lS6dTpdxVhb88/I6+Hs+438c3lfUow==", + "license": "MIT" + }, + "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/csstype": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", + "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", + "devOptional": true, + "license": "MIT" + }, + "node_modules/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-node-es": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/detect-node-es/-/detect-node-es-1.1.0.tgz", + "integrity": "sha512-ypdmJU/TbBby2Dxibuv7ZLW3Bs1QEmM7nHjEANfohJLvE0XVujisn1qPJcZxg+qDucsr+bP6fLD1rPS3AhJ7EQ==", + "license": "MIT" + }, + "node_modules/electron-to-chromium": { + "version": "1.5.353", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.353.tgz", + "integrity": "sha512-kOrWphBi8TOZyiJZqsgqIle0lw+tzmnQK83pV9dZUd01Nm2POECSyFQMAuarzZdYqQW7FH9RaYOuaRo3h+bQ3w==", + "dev": true, + "license": "ISC" + }, + "node_modules/esbuild": { + "version": "0.21.5", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", + "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.21.5", + "@esbuild/android-arm": "0.21.5", + "@esbuild/android-arm64": "0.21.5", + "@esbuild/android-x64": "0.21.5", + "@esbuild/darwin-arm64": "0.21.5", + "@esbuild/darwin-x64": "0.21.5", + "@esbuild/freebsd-arm64": "0.21.5", + "@esbuild/freebsd-x64": "0.21.5", + "@esbuild/linux-arm": "0.21.5", + "@esbuild/linux-arm64": "0.21.5", + "@esbuild/linux-ia32": "0.21.5", + "@esbuild/linux-loong64": "0.21.5", + "@esbuild/linux-mips64el": "0.21.5", + "@esbuild/linux-ppc64": "0.21.5", + "@esbuild/linux-riscv64": "0.21.5", + "@esbuild/linux-s390x": "0.21.5", + "@esbuild/linux-x64": "0.21.5", + "@esbuild/netbsd-x64": "0.21.5", + "@esbuild/openbsd-x64": "0.21.5", + "@esbuild/sunos-x64": "0.21.5", + "@esbuild/win32-arm64": "0.21.5", + "@esbuild/win32-ia32": "0.21.5", + "@esbuild/win32-x64": "0.21.5" + } + }, + "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/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/get-nonce": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-nonce/-/get-nonce-1.0.1.tgz", + "integrity": "sha512-FJhYRoDaiatfEkUK8HKlicmu/3SGFD51q3itKDGoSTysQJBnfOcxU5GxnhE1E6soB76MbT0MBtnKJuXyAx+96Q==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "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/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/markdown-to-jsx": { + "version": "9.8.1", + "resolved": "https://registry.npmjs.org/markdown-to-jsx/-/markdown-to-jsx-9.8.1.tgz", + "integrity": "sha512-yq70dLPkBnE2LYFtGTLfRes4qyBDS+a4wDttAA/b/BzVGrbs2e0TfCeSFrMkapCg1lsxYi+42BowuBDxLP9k4Q==", + "license": "MIT", + "engines": { + "node": ">= 18" + }, + "peerDependencies": { + "react": ">= 16.0.0", + "solid-js": ">=1.0.0", + "vue": ">=3.0.0" + }, + "peerDependenciesMeta": { + "react": { + "optional": true + }, + "react-native": { + "optional": true + }, + "solid-js": { + "optional": true + }, + "vue": { + "optional": true + } + } + }, + "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.12", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz", + "integrity": "sha512-ZB9RH/39qpq5Vu6Y+NmUaFhQR6pp+M2Xt76XBnEwDaGcVAqhlvxrl3B2bKS5D3NH3QR76v3aSrKaF/Kiy7lEtQ==", + "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.38", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.38.tgz", + "integrity": "sha512-3qT/88Y3FbH/Kx4szpQQ4HzUbVrHPKTLVpVocKiLfoYvw9XSGOX2FmD2d6DrXbVYyAQTF2HeF6My8jmzx7/CRw==", + "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/postcss": { + "version": "8.5.14", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.14.tgz", + "integrity": "sha512-SoSL4+OSEtR99LHFZQiJLkT59C5B1amGO1NzTwj7TT1qCUgUO6hxOvzkOYxD+vMrXBM3XJIKzokoERdqQq/Zmg==", + "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/radix-ui": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/radix-ui/-/radix-ui-1.4.3.tgz", + "integrity": "sha512-aWizCQiyeAenIdUbqEpXgRA1ya65P13NKn/W8rWkcN0OPkRDxdBVLWnIEDsS2RpwCK2nobI7oMUSmexzTDyAmA==", + "license": "MIT", + "dependencies": { + "@radix-ui/primitive": "1.1.3", + "@radix-ui/react-accessible-icon": "1.1.7", + "@radix-ui/react-accordion": "1.2.12", + "@radix-ui/react-alert-dialog": "1.1.15", + "@radix-ui/react-arrow": "1.1.7", + "@radix-ui/react-aspect-ratio": "1.1.7", + "@radix-ui/react-avatar": "1.1.10", + "@radix-ui/react-checkbox": "1.3.3", + "@radix-ui/react-collapsible": "1.1.12", + "@radix-ui/react-collection": "1.1.7", + "@radix-ui/react-compose-refs": "1.1.2", + "@radix-ui/react-context": "1.1.2", + "@radix-ui/react-context-menu": "2.2.16", + "@radix-ui/react-dialog": "1.1.15", + "@radix-ui/react-direction": "1.1.1", + "@radix-ui/react-dismissable-layer": "1.1.11", + "@radix-ui/react-dropdown-menu": "2.1.16", + "@radix-ui/react-focus-guards": "1.1.3", + "@radix-ui/react-focus-scope": "1.1.7", + "@radix-ui/react-form": "0.1.8", + "@radix-ui/react-hover-card": "1.1.15", + "@radix-ui/react-label": "2.1.7", + "@radix-ui/react-menu": "2.1.16", + "@radix-ui/react-menubar": "1.1.16", + "@radix-ui/react-navigation-menu": "1.2.14", + "@radix-ui/react-one-time-password-field": "0.1.8", + "@radix-ui/react-password-toggle-field": "0.1.3", + "@radix-ui/react-popover": "1.1.15", + "@radix-ui/react-popper": "1.2.8", + "@radix-ui/react-portal": "1.1.9", + "@radix-ui/react-presence": "1.1.5", + "@radix-ui/react-primitive": "2.1.3", + "@radix-ui/react-progress": "1.1.7", + "@radix-ui/react-radio-group": "1.3.8", + "@radix-ui/react-roving-focus": "1.1.11", + "@radix-ui/react-scroll-area": "1.2.10", + "@radix-ui/react-select": "2.2.6", + "@radix-ui/react-separator": "1.1.7", + "@radix-ui/react-slider": "1.3.6", + "@radix-ui/react-slot": "1.2.3", + "@radix-ui/react-switch": "1.2.6", + "@radix-ui/react-tabs": "1.1.13", + "@radix-ui/react-toast": "1.2.15", + "@radix-ui/react-toggle": "1.1.10", + "@radix-ui/react-toggle-group": "1.1.11", + "@radix-ui/react-toolbar": "1.1.11", + "@radix-ui/react-tooltip": "1.2.8", + "@radix-ui/react-use-callback-ref": "1.1.1", + "@radix-ui/react-use-controllable-state": "1.2.2", + "@radix-ui/react-use-effect-event": "0.0.2", + "@radix-ui/react-use-escape-keydown": "1.1.1", + "@radix-ui/react-use-is-hydrated": "0.1.0", + "@radix-ui/react-use-layout-effect": "1.1.1", + "@radix-ui/react-use-size": "1.1.1", + "@radix-ui/react-visually-hidden": "1.2.3" + }, + "peerDependencies": { + "@types/react": "*", + "@types/react-dom": "*", + "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", + "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, + "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", + "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", + "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-remove-scroll": { + "version": "2.7.2", + "resolved": "https://registry.npmjs.org/react-remove-scroll/-/react-remove-scroll-2.7.2.tgz", + "integrity": "sha512-Iqb9NjCCTt6Hf+vOdNIZGdTiH1QSqr27H/Ek9sv/a97gfueI/5h1s3yRi1nngzMUaOOToin5dI1dXKdXiF+u0Q==", + "license": "MIT", + "dependencies": { + "react-remove-scroll-bar": "^2.3.7", + "react-style-singleton": "^2.2.3", + "tslib": "^2.1.0", + "use-callback-ref": "^1.3.3", + "use-sidecar": "^1.1.3" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/react-remove-scroll-bar": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/react-remove-scroll-bar/-/react-remove-scroll-bar-2.3.8.tgz", + "integrity": "sha512-9r+yi9+mgU33AKcj6IbT9oRCO78WriSj6t/cF8DWBZJ9aOGPOTEDvdUDz1FwKim7QXWwmHqtdHnRJfhAxEG46Q==", + "license": "MIT", + "dependencies": { + "react-style-singleton": "^2.2.2", + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/react-style-singleton": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/react-style-singleton/-/react-style-singleton-2.2.3.tgz", + "integrity": "sha512-b6jSvxvVnyptAiLjbkWLE/lOnR4lfTtDAl+eUC7RZy+QQWc6wRzIV2CE6xBuMmDxc2qIihtDCZD5NPOFl7fRBQ==", + "license": "MIT", + "dependencies": { + "get-nonce": "^1.0.0", + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/rollup": { + "version": "4.60.3", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.60.3.tgz", + "integrity": "sha512-pAQK9HalE84QSm4Po3EmWIZPd3FnjkShVkiMlz1iligWYkWQ7wHYd1PF/T7QZ5TVSD6uSTon5gBVMSM4JfBV+A==", + "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.3", + "@rollup/rollup-android-arm64": "4.60.3", + "@rollup/rollup-darwin-arm64": "4.60.3", + "@rollup/rollup-darwin-x64": "4.60.3", + "@rollup/rollup-freebsd-arm64": "4.60.3", + "@rollup/rollup-freebsd-x64": "4.60.3", + "@rollup/rollup-linux-arm-gnueabihf": "4.60.3", + "@rollup/rollup-linux-arm-musleabihf": "4.60.3", + "@rollup/rollup-linux-arm64-gnu": "4.60.3", + "@rollup/rollup-linux-arm64-musl": "4.60.3", + "@rollup/rollup-linux-loong64-gnu": "4.60.3", + "@rollup/rollup-linux-loong64-musl": "4.60.3", + "@rollup/rollup-linux-ppc64-gnu": "4.60.3", + "@rollup/rollup-linux-ppc64-musl": "4.60.3", + "@rollup/rollup-linux-riscv64-gnu": "4.60.3", + "@rollup/rollup-linux-riscv64-musl": "4.60.3", + "@rollup/rollup-linux-s390x-gnu": "4.60.3", + "@rollup/rollup-linux-x64-gnu": "4.60.3", + "@rollup/rollup-linux-x64-musl": "4.60.3", + "@rollup/rollup-openbsd-x64": "4.60.3", + "@rollup/rollup-openharmony-arm64": "4.60.3", + "@rollup/rollup-win32-arm64-msvc": "4.60.3", + "@rollup/rollup-win32-ia32-msvc": "4.60.3", + "@rollup/rollup-win32-x64-gnu": "4.60.3", + "@rollup/rollup-win32-x64-msvc": "4.60.3", + "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/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/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "license": "0BSD" + }, + "node_modules/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/use-callback-ref": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/use-callback-ref/-/use-callback-ref-1.3.3.tgz", + "integrity": "sha512-jQL3lRnocaFtu3V00JToYz/4QkNWswxijDaCVNZRiRTO3HQDLsdu1ZtmIUvV4yPp+rvWm5j0y0TG/S61cuijTg==", + "license": "MIT", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/use-sidecar": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/use-sidecar/-/use-sidecar-1.1.3.tgz", + "integrity": "sha512-Fedw0aZvkhynoPYlA5WXrMCAMm+nSWdZt6lzJQ7Ok8S6Q+VsHmHpRWndVRJ8Be0ZbkfPc5LRYH+5XrzXcEeLRQ==", + "license": "MIT", + "dependencies": { + "detect-node-es": "^1.1.0", + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "@types/react": "*", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/use-sync-external-store": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.6.0.tgz", + "integrity": "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w==", + "license": "MIT", + "peerDependencies": { + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" + } + }, + "node_modules/vite": { + "version": "5.4.21", + "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.21.tgz", + "integrity": "sha512-o5a9xKjbtuhY6Bi5S3+HvbRERmouabWbyUcpXXUA1u+GNUKoROi9byOJ8M0nHbHYHkYICiMlqxkg1KkYmm25Sw==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "^0.21.3", + "postcss": "^8.4.43", + "rollup": "^4.20.0" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^18.0.0 || >=20.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", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "sass-embedded": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.4.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "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/docs/reference-repos/adainrivers/app/package.json b/docs/reference-repos/adainrivers/app/package.json new file mode 100644 index 0000000..db0da88 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/package.json @@ -0,0 +1,32 @@ +{ + "name": "dune-dedicated-server-manager-app", + "private": true, + "version": "0.3.16", + "type": "module", + "scripts": { + "dev": "vite --host 127.0.0.1 --port 1420", + "build": "tsc && vite build", + "preview": "vite preview --host 127.0.0.1 --port 1420", + "tauri": "tauri" + }, + "dependencies": { + "@radix-ui/react-icons": "^1.3.2", + "@radix-ui/themes": "^3.2.1", + "@tauri-apps/api": "^2.0.0", + "@tauri-apps/plugin-dialog": "^2.7.1", + "@tauri-apps/plugin-process": "^2.3.1", + "@tauri-apps/plugin-shell": "^2.3.5", + "@tauri-apps/plugin-updater": "^2.10.1", + "markdown-to-jsx": "^9.8.1", + "react": "^18.3.1", + "react-dom": "^18.3.1" + }, + "devDependencies": { + "@tauri-apps/cli": "^2.0.0", + "@types/react": "^18.3.12", + "@types/react-dom": "^18.3.1", + "@vitejs/plugin-react": "^4.3.3", + "typescript": "^5.6.3", + "vite": "^5.4.10" + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/Cargo.toml b/docs/reference-repos/adainrivers/app/src-tauri/Cargo.toml new file mode 100644 index 0000000..50b04d8 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/Cargo.toml @@ -0,0 +1,26 @@ +[package] +name = "dune-dedicated-server-manager-app" +version = "0.2.0" +description = "Desktop shell for Dune Dedicated Server Manager" +authors = ["Dune Dedicated Server Manager"] +edition = "2021" + +[lib] +name = "dune_dedicated_server_manager_app_lib" +crate-type = ["staticlib", "cdylib", "rlib"] + +[build-dependencies] +tauri-build = { version = "2", features = [] } + +[dependencies] +dune-manager-core = { path = "../../crates/dune-manager-core" } +tauri = { version = "2", features = ["devtools"] } +serde = { workspace = true } +serde_json = { workspace = true } +tauri-plugin-dialog = "2" +tauri-plugin-updater = "2" +tauri-plugin-process = "2" +tauri-plugin-shell = "2" +base64 = "0.22" +chrono = { version = "0.4", default-features = false, features = ["clock", "std"] } +reqwest = { version = "0.12", default-features = false, features = ["json"] } diff --git a/docs/reference-repos/adainrivers/app/src-tauri/binaries/.gitignore b/docs/reference-repos/adainrivers/app/src-tauri/binaries/.gitignore new file mode 100644 index 0000000..4f2dee6 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/binaries/.gitignore @@ -0,0 +1,6 @@ +# Populated by CI from the `linux-service-binary` job artifact, or locally +# via `cargo zigbuild -p dune-server-service --release --target +# x86_64-unknown-linux-musl` + manual copy. Not tracked. +dune-server-service +dune-server-service.service +dune-server-service.openrc diff --git a/docs/reference-repos/adainrivers/app/src-tauri/binaries/README.md b/docs/reference-repos/adainrivers/app/src-tauri/binaries/README.md new file mode 100644 index 0000000..51b1cba --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/binaries/README.md @@ -0,0 +1,23 @@ +# Bundled service binaries + +This directory holds the Linux `dune-server-service` binary (musl-static), its +systemd unit, and its OpenRC init script. They are populated by the +`linux-service-binary` job in `.github/workflows/release.yml` and bundled into +the desktop installer as Tauri resources. + +For local debug builds the directory can be empty — the `install_management_service` +Tauri command surfaces a friendly error when the resource is missing. + +For a local end-to-end test, build the service yourself: + +```powershell +rustup target add x86_64-unknown-linux-musl +cargo install --locked cargo-zigbuild +cargo zigbuild -p dune-server-service --release --target x86_64-unknown-linux-musl +Copy-Item target\x86_64-unknown-linux-musl\release\dune-server-service ` + app\src-tauri\binaries\dune-server-service +Copy-Item crates\dune-server-service\systemd\dune-server-service.service ` + app\src-tauri\binaries\dune-server-service.service +Copy-Item crates\dune-server-service\openrc\dune-server-service ` + app\src-tauri\binaries\dune-server-service.openrc +``` diff --git a/docs/reference-repos/adainrivers/app/src-tauri/build.rs b/docs/reference-repos/adainrivers/app/src-tauri/build.rs new file mode 100644 index 0000000..a74988d --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/build.rs @@ -0,0 +1,67 @@ +fn main() { + expose_dune_server_service_version(); + rerun_if_bundled_binaries_change(); + tauri_build::build(); +} + +/// Tauri's resource-copy step only fires when Cargo decides build.rs needs to +/// re-run, which by default doesn't watch arbitrary files. Without these +/// `rerun-if-changed` lines, refreshing the bundled `dune-server-service` +/// binary or its systemd/openrc units in `binaries/` after a previous build +/// produces a stale `target/release/binaries/` copy — the running exe then +/// pushes the OLD binary on Install/Update, with no visible signal. +fn rerun_if_bundled_binaries_change() { + let dir = std::path::Path::new(env!("CARGO_MANIFEST_DIR")).join("binaries"); + // Watch the directory itself so file additions/deletions also trigger a rerun. + println!("cargo:rerun-if-changed={}", dir.display()); + if let Ok(entries) = std::fs::read_dir(&dir) { + for entry in entries.flatten() { + let path = entry.path(); + // Skip README, .gitignore, and similar bookkeeping files. + if matches!( + path.file_name().and_then(|n| n.to_str()), + Some("README.md") | Some(".gitignore") + ) { + continue; + } + println!("cargo:rerun-if-changed={}", path.display()); + } + } +} + +fn expose_dune_server_service_version() { + let cargo_toml = std::path::Path::new(env!("CARGO_MANIFEST_DIR")) + .join("../../crates/dune-server-service/Cargo.toml"); + println!("cargo:rerun-if-changed={}", cargo_toml.display()); + let contents = std::fs::read_to_string(&cargo_toml) + .unwrap_or_else(|err| panic!("reading {}: {err}", cargo_toml.display())); + let version = parse_package_version(&contents).unwrap_or_else(|| { + panic!( + "could not find [package].version in {}", + cargo_toml.display() + ) + }); + println!("cargo:rustc-env=DUNE_SERVER_SERVICE_VERSION={version}"); +} + +fn parse_package_version(toml: &str) -> Option { + let mut in_package = false; + for line in toml.lines() { + let trimmed = line.trim(); + if trimmed.starts_with('[') { + in_package = trimmed == "[package]"; + continue; + } + if !in_package { + continue; + } + if let Some(rest) = trimmed.strip_prefix("version") { + let rest = rest.trim_start(); + let rest = rest.strip_prefix('=')?.trim_start(); + let rest = rest.trim_start_matches('"'); + let end = rest.find('"')?; + return Some(rest[..end].to_string()); + } + } + None +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/capabilities/default.json b/docs/reference-repos/adainrivers/app/src-tauri/capabilities/default.json new file mode 100644 index 0000000..ea6f606 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/capabilities/default.json @@ -0,0 +1,7 @@ +{ + "$schema": "../gen/schemas/desktop-schema.json", + "identifier": "default", + "description": "Default desktop app permissions", + "windows": ["main"], + "permissions": ["core:default", "dialog:allow-open", "process:default", "shell:allow-open", "updater:default"] +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/128x128.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/128x128.png new file mode 100644 index 0000000..809f78f Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/128x128.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/128x128@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/128x128@2x.png new file mode 100644 index 0000000..8dcc9d5 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/128x128@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/32x32.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/32x32.png new file mode 100644 index 0000000..8842930 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/32x32.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/64x64.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/64x64.png new file mode 100644 index 0000000..223e02c Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/64x64.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square107x107Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square107x107Logo.png new file mode 100644 index 0000000..72fe760 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square107x107Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square142x142Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square142x142Logo.png new file mode 100644 index 0000000..5e2dcb4 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square142x142Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square150x150Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square150x150Logo.png new file mode 100644 index 0000000..22ea41b Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square150x150Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square284x284Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square284x284Logo.png new file mode 100644 index 0000000..fce3974 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square284x284Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square30x30Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square30x30Logo.png new file mode 100644 index 0000000..9bf3e6e Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square30x30Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square310x310Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square310x310Logo.png new file mode 100644 index 0000000..44c5c53 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square310x310Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square44x44Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square44x44Logo.png new file mode 100644 index 0000000..3cd69af Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square44x44Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square71x71Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square71x71Logo.png new file mode 100644 index 0000000..526f4e4 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square71x71Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/Square89x89Logo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square89x89Logo.png new file mode 100644 index 0000000..77f5419 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/Square89x89Logo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/StoreLogo.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/StoreLogo.png new file mode 100644 index 0000000..6636775 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/StoreLogo.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml new file mode 100644 index 0000000..2ffbf24 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher.png new file mode 100644 index 0000000..3770800 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher_foreground.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..1d1c833 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher_foreground.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher_round.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher_round.png new file mode 100644 index 0000000..a5c5e6d Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-hdpi/ic_launcher_round.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher.png new file mode 100644 index 0000000..fe9b827 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher_foreground.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..d696169 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher_foreground.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher_round.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher_round.png new file mode 100644 index 0000000..c323886 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-mdpi/ic_launcher_round.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher.png new file mode 100644 index 0000000..9a9d9cf Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_foreground.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..e573c97 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_foreground.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_round.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_round.png new file mode 100644 index 0000000..7fae697 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_round.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher.png new file mode 100644 index 0000000..c1f1b73 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_foreground.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..f86c9b7 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_foreground.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_round.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_round.png new file mode 100644 index 0000000..31dc0a8 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_round.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 0000000..c4434f7 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_foreground.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_foreground.png new file mode 100644 index 0000000..32746bf Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_foreground.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_round.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_round.png new file mode 100644 index 0000000..6e05ee9 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_round.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/android/values/ic_launcher_background.xml b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/values/ic_launcher_background.xml new file mode 100644 index 0000000..ea9c223 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/icons/android/values/ic_launcher_background.xml @@ -0,0 +1,4 @@ + + + #fff + \ No newline at end of file diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/icon.ico b/docs/reference-repos/adainrivers/app/src-tauri/icons/icon.ico new file mode 100644 index 0000000..789df49 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/icon.ico differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/icon.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/icon.png new file mode 100644 index 0000000..6a90b0a Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/icon.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@1x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@1x.png new file mode 100644 index 0000000..68ff6b9 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@1x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@2x-1.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@2x-1.png new file mode 100644 index 0000000..aa6e0e1 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@2x-1.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@2x.png new file mode 100644 index 0000000..aa6e0e1 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@3x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@3x.png new file mode 100644 index 0000000..25f4b74 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-20x20@3x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@1x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@1x.png new file mode 100644 index 0000000..83a478d Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@1x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@2x-1.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@2x-1.png new file mode 100644 index 0000000..60e9fe3 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@2x-1.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@2x.png new file mode 100644 index 0000000..60e9fe3 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@3x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@3x.png new file mode 100644 index 0000000..1e06c26 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-29x29@3x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@1x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@1x.png new file mode 100644 index 0000000..aa6e0e1 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@1x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@2x-1.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@2x-1.png new file mode 100644 index 0000000..32dfebc Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@2x-1.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@2x.png new file mode 100644 index 0000000..32dfebc Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@3x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@3x.png new file mode 100644 index 0000000..c828066 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-40x40@3x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-512@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-512@2x.png new file mode 100644 index 0000000..d7f13a4 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-512@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-60x60@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-60x60@2x.png new file mode 100644 index 0000000..c828066 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-60x60@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-60x60@3x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-60x60@3x.png new file mode 100644 index 0000000..d66d187 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-60x60@3x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-76x76@1x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-76x76@1x.png new file mode 100644 index 0000000..177586c Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-76x76@1x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-76x76@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-76x76@2x.png new file mode 100644 index 0000000..1f03f5b Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-76x76@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-83.5x83.5@2x.png b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-83.5x83.5@2x.png new file mode 100644 index 0000000..0b022f6 Binary files /dev/null and b/docs/reference-repos/adainrivers/app/src-tauri/icons/ios/AppIcon-83.5x83.5@2x.png differ diff --git a/docs/reference-repos/adainrivers/app/src-tauri/resources/manager-api/.gitkeep b/docs/reference-repos/adainrivers/app/src-tauri/resources/manager-api/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/battlegroup.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/battlegroup.rs new file mode 100644 index 0000000..c388bea --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/battlegroup.rs @@ -0,0 +1,254 @@ +use dune_manager_core::orchestration::{ + is_started_state, BattlegroupManagementOrchestrator, BattlegroupRef, BattlegroupState, + RusshRunner, StructuredKubectl, VendorBattlegroupWrapper, +}; + +use crate::commands::shared::{command_error_message, runner_for_remote_kind}; +use crate::commands::status_data::read_remote_server_status; +use crate::dto::{RemoteBattlegroupStatus, RemoteServerActionRequest, RemoteServerStatus}; +use crate::logging::TauriOperationSink; + +type Manager = BattlegroupManagementOrchestrator< + StructuredKubectl, + VendorBattlegroupWrapper, +>; + +fn manager_from_runner(runner: &RusshRunner) -> Manager { + let kubernetes = StructuredKubectl::new(runner.clone()); + // Pass the actual SSH login user so the wrapper knows when to insert + // `sudo -n -u dune -H bash -lc ...`. Defaulting to "dune" here was a + // silent root-style fallback: when the operator registered the server + // under e.g. `ubuntu`, the wrapper skipped impersonation and the script + // tried to read/write /home/dune as ubuntu, which fails noisily. + let ssh_user = runner.target().user.clone(); + let wrapper = VendorBattlegroupWrapper::with_ssh_user(runner.clone(), ssh_user); + BattlegroupManagementOrchestrator::new(kubernetes, wrapper) +} + +#[tauri::command] +pub async fn start_remote_battlegroup( + app: tauri::AppHandle, + request: RemoteServerActionRequest, +) -> Result { + run_remote_battlegroup_action(app, request, false).await +} + +#[tauri::command] +pub async fn stop_remote_battlegroup( + app: tauri::AppHandle, + request: RemoteServerActionRequest, +) -> Result { + run_remote_battlegroup_action(app, request, true).await +} + +#[tauri::command] +pub async fn restart_remote_battlegroup( + app: tauri::AppHandle, + request: RemoteServerActionRequest, +) -> Result { + let worker_app = app.clone(); + tauri::async_runtime::spawn_blocking(move || { + let mut sink = TauriOperationSink::new(worker_app); + sink.info("bg.restart", "Restarting remote battlegroup."); + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host, + request.user, + request.key_path, + Some(request.port), + )?; + let battlegroup = BattlegroupRef { + namespace: request.namespace, + name: request.battlegroup_name, + }; + let manager = manager_from_runner(&runner); + manager + .restart_and_wait_director(&battlegroup, 240, &mut sink) + .map_err(command_error_message)?; + sink.info("bg.restart", "Refreshing battlegroup state."); + read_remote_server_status(&runner, &battlegroup.namespace, &battlegroup.name) + .map_err(command_error_message) + }) + .await + .map_err(|err| format!("Remote battlegroup restart worker failed: {err}"))? +} + +#[tauri::command] +pub async fn update_remote_battlegroup( + app: tauri::AppHandle, + request: RemoteServerActionRequest, +) -> Result { + let worker_app = app.clone(); + tauri::async_runtime::spawn_blocking(move || { + let mut sink = TauriOperationSink::new(worker_app); + sink.info("bg.update", "Running vendor wrapper update."); + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host, + request.user, + request.key_path, + Some(request.port), + )?; + run_battlegroup_update_with_runner( + &runner, + &mut sink, + request.namespace, + request.battlegroup_name, + ) + }) + .await + .map_err(|err| format!("Remote battlegroup update worker failed: {err}"))? +} + +pub async fn run_remote_battlegroup_action( + app: tauri::AppHandle, + request: RemoteServerActionRequest, + stop: bool, +) -> Result { + let worker_app = app.clone(); + tauri::async_runtime::spawn_blocking(move || { + let mut sink = TauriOperationSink::new(worker_app); + sink.info("bg.check", "Checking remote battlegroup state."); + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host, + request.user, + request.key_path, + Some(request.port), + )?; + run_battlegroup_action_with_runner( + &runner, + &mut sink, + request.namespace, + request.battlegroup_name, + stop, + ) + }) + .await + .map_err(|err| format!("Remote battlegroup action worker failed: {err}"))? +} + +fn run_battlegroup_action_with_runner( + runner: &RusshRunner, + sink: &mut TauriOperationSink, + namespace: String, + battlegroup_name: String, + stop: bool, +) -> Result { + let battlegroup = BattlegroupRef { + namespace, + name: battlegroup_name, + }; + let manager = manager_from_runner(runner); + // Pre-flight no-op guard. Read the BattleGroup state from the stable + // kubectl JSON schema (same source as the dashboard) rather than the + // vendor wrapper's `status` text: that text layout drifts across Funcom + // releases and was being misparsed into bogus phases (e.g. status="World", + // director="2/2"), which made `is_started_state` wrongly report the BG as + // not running and refuse a perfectly valid Stop (#19). + let before = read_remote_server_status(runner, &battlegroup.namespace, &battlegroup.name) + .map_err(command_error_message)?; + let before_bg = &before.battlegroup; + let before_started = is_started_state(&battlegroup_state_from_status(before_bg)); + if stop && !before_started { + return Err(format!( + "Battlegroup is not running (status={}, stop={}, database={}, gateway={}, director={}).", + before_bg.phase, + before_bg.stop, + before_bg.database_phase, + before_bg.server_group_phase, + before_bg.director_phase + )); + } + if !stop && before_started { + return Err("Battlegroup is already started.".to_string()); + } + if stop { + manager + .stop(&battlegroup, sink) + .map_err(command_error_message)?; + } else { + manager + .start_and_wait_director(&battlegroup, 180, sink) + .map_err(command_error_message)?; + } + sink.info("bg.check", "Refreshing battlegroup state."); + read_remote_server_status(runner, &battlegroup.namespace, &battlegroup.name) + .map_err(command_error_message) +} + +/// Adapts the structured `RemoteBattlegroupStatus` (read from the BattleGroup +/// CR JSON) into the core `BattlegroupState` so the shared `is_started_state` +/// phase vocabulary stays the single source of truth. `server_stats` is not +/// consulted by `is_started_state`, so it is left empty. +fn battlegroup_state_from_status(status: &RemoteBattlegroupStatus) -> BattlegroupState { + BattlegroupState { + stop: status.stop, + phase: status.phase.clone(), + database_phase: status.database_phase.clone(), + server_group_phase: status.server_group_phase.clone(), + director_phase: status.director_phase.clone(), + uptime: status.uptime.clone(), + server_stats: Vec::new(), + } +} + +fn run_battlegroup_update_with_runner( + runner: &RusshRunner, + sink: &mut TauriOperationSink, + namespace: String, + battlegroup_name: String, +) -> Result { + let battlegroup = BattlegroupRef { + namespace, + name: battlegroup_name, + }; + let manager = manager_from_runner(runner); + sink.warn( + "bg.update", + "Running vendor `battlegroup update` (steamcmd + operators + maps + images).", + ); + let stdout = manager + .update(&battlegroup, sink) + .map_err(command_error_message)?; + if !stdout.trim().is_empty() { + sink.info("bg.update", stdout.trim().to_string()); + } + sink.info("bg.update", "Refreshing battlegroup state."); + read_remote_server_status(runner, &battlegroup.namespace, &battlegroup.name) + .map_err(command_error_message) +} + +#[cfg(test)] +mod tests { + use super::*; + + fn status(phase: &str, sgp: &str, director: &str, stop: bool) -> RemoteBattlegroupStatus { + RemoteBattlegroupStatus { + stop, + phase: phase.to_string(), + database_phase: "Ready".to_string(), + server_group_phase: sgp.to_string(), + director_phase: director.to_string(), + uptime: "8h45m".to_string(), + server_stats: Vec::new(), + } + } + + #[test] + fn reconciling_bg_counts_as_started_so_stop_is_allowed() { + // #19: the structured kubectl read reports phase=Reconciling, + // serverGroupPhase=Running, directorPhase=Healthy while the BG is up. + // The stop guard must treat this as started (previously the wrapper + // text-parse produced status="World"/director="2/2" and refused). + let s = status("Reconciling", "Running", "Healthy", false); + assert!(is_started_state(&battlegroup_state_from_status(&s))); + } + + #[test] + fn stopped_bg_is_not_started() { + assert!(!is_started_state(&battlegroup_state_from_status(&status( + "Stopped", "Stopped", "", true + )))); + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/component.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/component.rs new file mode 100644 index 0000000..2a3e981 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/component.rs @@ -0,0 +1,168 @@ +use dune_manager_core::models::CommandResult; +use dune_manager_core::orchestration::{RemoteCommandRunner, RusshRunner}; +use dune_manager_core::security::redact_text; + +use crate::commands::shared::{command_error_message, runner_for_remote_kind, sh_single_quoted}; +use crate::dto::{ + RemoteComponentLogRequest, RemoteComponentLogResult, RemoteComponentRestartRequest, + RemoteComponentRestartResult, +}; + +#[tauri::command] +pub async fn remote_component_log_tail( + request: RemoteComponentLogRequest, +) -> Result { + tauri::async_runtime::spawn_blocking(move || { + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host, + request.user, + request.key_path, + Some(request.port), + )?; + read_remote_component_log_tail( + &runner, + &request.namespace, + &request.component, + request.tail, + ) + .map_err(command_error_message) + }) + .await + .map_err(|err| format!("Remote component log worker failed: {err}"))? +} + +#[tauri::command] +pub async fn restart_remote_component( + request: RemoteComponentRestartRequest, +) -> Result { + tauri::async_runtime::spawn_blocking(move || { + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host, + request.user, + request.key_path, + Some(request.port), + )?; + restart_remote_component_inner(&runner, &request.namespace, &request.component) + .map_err(command_error_message) + }) + .await + .map_err(|err| format!("Remote component restart worker failed: {err}"))? +} + +fn read_remote_component_log_tail( + runner: &RusshRunner, + namespace: &str, + component: &str, + tail: u32, +) -> CommandResult { + let component = component.trim(); + let (mode, pattern) = component_pod_selection(component)?; + let tail = tail.clamp(20, 500); + let script = format!( + r#" +ns={ns} +mode={mode} +pattern={pattern} +tail_lines={tail} +component={component} + +if [ "$mode" = "role" ]; then + pods=$(sudo kubectl get pods -n "$ns" -l "role=$pattern" --no-headers -o custom-columns=NAME:.metadata.name 2>/dev/null || true) +elif [ "$mode" = "roles" ]; then + pods=$(sudo kubectl get pods -n "$ns" --no-headers -o custom-columns=NAME:.metadata.name,ROLE:.metadata.labels.role 2>/dev/null | grep -E "$pattern" | awk '{{print $1}}' || true) +else + pods=$(sudo kubectl get pods -n "$ns" --no-headers -o custom-columns=NAME:.metadata.name 2>/dev/null | grep -- "$pattern" || true) +fi + +if [ -z "$pods" ]; then + echo "No pods found for $component." + exit 0 +fi + +for pod in $pods; do + echo "== $pod ==" + sudo kubectl logs -n "$ns" "$pod" --all-containers --tail="$tail_lines" 2>&1 || true +done +"#, + ns = sh_single_quoted(namespace), + mode = sh_single_quoted(mode), + pattern = sh_single_quoted(pattern), + tail = tail, + component = sh_single_quoted(component), + ); + let output = runner.run_script(&script)?; + Ok(RemoteComponentLogResult { + component: component.to_string(), + output: redact_text(&output), + }) +} + +fn restart_remote_component_inner( + runner: &RusshRunner, + namespace: &str, + component: &str, +) -> CommandResult { + let component = component.trim(); + let (mode, pattern) = component_pod_selection(component)?; + let script = format!( + r#" +ns={ns} +mode={mode} +pattern={pattern} +component={component} + +if [ "$mode" = "role" ]; then + pods=$(sudo kubectl get pods -n "$ns" -l "role=$pattern" --no-headers -o custom-columns=NAME:.metadata.name 2>/dev/null || true) +elif [ "$mode" = "roles" ]; then + pods=$(sudo kubectl get pods -n "$ns" --no-headers -o custom-columns=NAME:.metadata.name,ROLE:.metadata.labels.role 2>/dev/null | grep -E "$pattern" | awk '{{print $1}}' || true) +else + pods=$(sudo kubectl get pods -n "$ns" --no-headers -o custom-columns=NAME:.metadata.name 2>/dev/null | grep -- "$pattern" || true) +fi + +if [ -z "$pods" ]; then + echo "No pods found for $component." + exit 0 +fi + +for pod in $pods; do + echo "Restarting $pod" + sudo kubectl delete pod -n "$ns" "$pod" --wait=false +done +"#, + ns = sh_single_quoted(namespace), + mode = sh_single_quoted(mode), + pattern = sh_single_quoted(pattern), + component = sh_single_quoted(component), + ); + let output = runner.run_script(&script)?; + Ok(RemoteComponentRestartResult { + component: component.to_string(), + output: redact_text(&output), + }) +} + +fn component_pod_selection(component: &str) -> CommandResult<(&'static str, &'static str)> { + match component { + "database" => Ok(("role", "igw-database")), + "database-utilities" => Ok(( + "roles", + "igw-database-utility|igw-database-monitor|igw-database-pghero", + )), + "message-queue" => Ok(("role", "igw-message-queue")), + "director" => Ok(("role", "igw-battlegroup-director")), + "gateway" | "gateway-resource" => Ok(("role", "igw-server-gateway")), + "text-router" => Ok(("role", "igw-text-router")), + "file-browser" => Ok(("role", "igw-filebrowser")), + "server-group" => Ok(("role", "igw-server")), + "map-survival-1" => Ok(("name", "-sg-survival-1-")), + "map-overmap" => Ok(("name", "-sg-overmap-")), + "map-deepdesert" => Ok(("name", "-sg-deepdesert-")), + "map-social-arrakeen" => Ok(("name", "-sg-sh-arrakeen-")), + "map-social-harkovillage" => Ok(("name", "-sg-sh-harkovillage-")), + _ => Err(dune_manager_core::errors::failure(format!( + "Unknown component key: {component}" + ))), + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/discovery.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/discovery.rs new file mode 100644 index 0000000..65ad83a --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/discovery.rs @@ -0,0 +1,34 @@ +use dune_manager_core::orchestration::RemoteCommandRunner; + +use crate::commands::shared::{command_error_message, runner_for_remote_kind}; +use crate::commands::status_data::remote_records_from_battlegroups; +use crate::dto::{RemoteConnectionRequest, RemoteServerRecord}; + +#[tauri::command] +pub async fn detect_remote_ubuntu_servers( + request: RemoteConnectionRequest, +) -> Result, String> { + tauri::async_runtime::spawn_blocking(move || { + let request = RemoteConnectionRequest { + server_type: Some("ubuntu".to_string()), + ..request + }; + let user = request.user.clone().unwrap_or_default(); + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host.clone(), + user, + request.key_path.clone(), + Some(request.port), + )?; + let value = runner + .run_json( + "sudo kubectl get battlegroups -A -o json", + "remote ubuntu battlegroups", + ) + .map_err(command_error_message)?; + Ok(remote_records_from_battlegroups(&request, &value)) + }) + .await + .map_err(|err| format!("Remote server detection worker failed: {err}"))? +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/logs.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/logs.rs new file mode 100644 index 0000000..9c92def --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/logs.rs @@ -0,0 +1,36 @@ +//! Frontend-facing helpers for the persisted operation log file. + +use std::sync::Arc; + +use tauri::State; + +use crate::log_file::LogFile; + +/// Appends a single row to the persisted operation log. +/// +/// Frontend-originated log rows (those produced directly by React without a +/// matching Rust event) call this so the on-disk log mirrors the in-memory +/// view exactly. +#[tauri::command] +pub fn record_operation_log( + log_file: State<'_, Arc>, + level: String, + scope: String, + message: String, +) -> Result<(), String> { + let allowed_levels = ["debug", "info", "warn", "error"]; + let normalized = if allowed_levels.contains(&level.as_str()) { + level.as_str() + } else { + "info" + }; + log_file + .append(normalized, &scope, &message) + .map_err(|err| err.to_string()) +} + +/// Returns the absolute path of the directory containing operation.log. +#[tauri::command] +pub fn get_logs_folder(log_file: State<'_, Arc>) -> String { + log_file.dir().to_string_lossy().into_owned() +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/management_api.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/management_api.rs new file mode 100644 index 0000000..0598703 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/management_api.rs @@ -0,0 +1,490 @@ +use std::time::Duration; + +use reqwest::Client; +use serde_json::Value; +use tauri::Manager; + +use crate::state::TunnelRegistry; + +pub fn ensure_client(app: &tauri::AppHandle) -> Client { + if let Some(client) = app.try_state::() { + return client.inner().clone(); + } + let client = Client::builder() + .timeout(Duration::from_secs(20)) + .build() + .expect("reqwest client builds"); + app.manage(client.clone()); + client +} + +fn tunnel_local_port(registry: &TunnelRegistry, tunnel_id: &str) -> Result { + let tunnels = registry + .tunnels + .lock() + .map_err(|_| "tunnel registry unavailable".to_string())?; + let tunnel = tunnels + .get(tunnel_id.trim()) + .ok_or_else(|| format!("no active tunnel id={tunnel_id}"))?; + Ok(tunnel.status.local_port) +} + +async fn get_json(client: &Client, port: u16, path: &str) -> Result { + let url = format!("http://127.0.0.1:{port}{path}"); + let resp = client + .get(&url) + .send() + .await + .map_err(|err| format!("GET {path}: {err}"))?; + if !resp.status().is_success() { + let status = resp.status(); + let body_text = resp.text().await.unwrap_or_default(); + return Err(format!("GET {path} -> {status}: {body_text}")); + } + resp.json::() + .await + .map_err(|err| format!("decoding {path}: {err}")) +} + +async fn post_json(client: &Client, port: u16, path: &str, body: &Value) -> Result { + let url = format!("http://127.0.0.1:{port}{path}"); + let resp = client + .post(&url) + .json(body) + .send() + .await + .map_err(|err| format!("POST {path}: {err}"))?; + if !resp.status().is_success() { + let status = resp.status(); + let body_text = resp.text().await.unwrap_or_default(); + return Err(format!("POST {path} -> {status}: {body_text}")); + } + resp.json::() + .await + .map_err(|err| format!("decoding {path}: {err}")) +} + +#[tauri::command] +pub async fn ms_health( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json(&client, port, "/api/health").await +} + +#[tauri::command] +pub async fn ms_list_runs( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + limit: Option, + task: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let mut path = String::from("/api/runs"); + let mut sep = '?'; + if let Some(l) = limit { + path.push(sep); + path.push_str(&format!("limit={l}")); + sep = '&'; + } + if let Some(t) = task { + path.push(sep); + path.push_str(&format!("task={t}")); + } + get_json(&client, port, &path).await +} + +#[tauri::command] +pub async fn ms_list_logs( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + limit: Option, + run_id: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let mut path = String::from("/api/logs"); + let mut sep = '?'; + if let Some(l) = limit { + path.push(sep); + path.push_str(&format!("limit={l}")); + sep = '&'; + } + if let Some(r) = run_id { + path.push(sep); + path.push_str(&format!("runId={r}")); + } + get_json(&client, port, &path).await +} + +#[tauri::command] +pub async fn ms_trigger_run( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + task: String, + options: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let mut body = serde_json::Map::new(); + body.insert("task".to_string(), Value::String(task)); + if let Some(opts) = options { + body.insert("options".to_string(), opts); + } + post_json(&client, port, "/api/runs/trigger", &Value::Object(body)).await +} + +#[tauri::command] +pub async fn ms_list_commands( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json(&client, port, "/api/admin/commands").await +} + +#[tauri::command] +pub async fn ms_search_items( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + q: Option, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json( + &client, + port, + &search_path("/api/admin/items", q.as_deref(), limit), + ) + .await +} + +#[tauri::command] +pub async fn ms_search_vehicles( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + q: Option, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json( + &client, + port, + &search_path("/api/admin/vehicles", q.as_deref(), limit), + ) + .await +} + +#[tauri::command] +pub async fn ms_search_skill_modules( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + q: Option, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json( + &client, + port, + &search_path("/api/admin/skill-modules", q.as_deref(), limit), + ) + .await +} + +#[tauri::command] +pub async fn ms_search_journey_nodes( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + q: Option, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json( + &client, + port, + &search_path("/api/admin/journey-nodes", q.as_deref(), limit), + ) + .await +} + +#[tauri::command] +pub async fn ms_search_xp_event_tags( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + q: Option, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json( + &client, + port, + &search_path("/api/admin/xp-event-tags", q.as_deref(), limit), + ) + .await +} + +#[tauri::command] +pub async fn ms_search_players( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + q: Option, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json( + &client, + port, + &search_path("/api/admin/players", q.as_deref(), limit), + ) + .await +} + +#[tauri::command] +pub async fn ms_cluster( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json(&client, port, "/api/admin/cluster").await +} + +#[tauri::command] +pub async fn ms_player_location( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + fls_id: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let path = format!("/api/admin/player-location?flsId={}", urlencoding(&fls_id)); + get_json(&client, port, &path).await +} + +#[tauri::command] +pub async fn ms_get_config( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json(&client, port, "/api/config").await +} + +#[tauri::command] +pub async fn ms_set_config( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + config: Value, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + post_json(&client, port, "/api/config", &config).await +} + +#[tauri::command] +pub async fn ms_list_timezones( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json(&client, port, "/api/timezones").await +} + +#[tauri::command] +pub async fn ms_cron_preview( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + expr: String, + count: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let mut path = format!("/api/cron/preview?expr={}", urlencoding(&expr)); + if let Some(c) = count { + path.push_str(&format!("&count={c}")); + } + get_json(&client, port, &path).await +} + +#[tauri::command] +pub async fn ms_dump_prune_preview( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + get_json(&client, port, "/api/maintenance/dump-prune").await +} + +#[tauri::command] +pub async fn ms_dump_prune_execute( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + items: Value, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let body = serde_json::json!({ "items": items }); + post_json(&client, port, "/api/maintenance/dump-prune", &body).await +} + +#[tauri::command] +pub async fn ms_history( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let path = match limit { + Some(l) => format!("/api/admin/history?limit={l}"), + None => String::from("/api/admin/history"), + }; + get_json(&client, port, &path).await +} + +#[tauri::command] +pub async fn ms_welcome_grants( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + limit: Option, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + let path = match limit { + Some(l) => format!("/api/admin/welcome-grants?limit={l}"), + None => String::from("/api/admin/welcome-grants"), + }; + get_json(&client, port, &path).await +} + +#[tauri::command] +pub async fn ms_welcome_grant_retry( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + player_id: String, + package_version: String, + account_id: i64, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + post_json( + &client, + port, + "/api/admin/welcome-grants/retry", + &serde_json::json!({ + "playerId": player_id, + "packageVersion": package_version, + "accountId": account_id, + }), + ) + .await +} + +#[tauri::command] +pub async fn ms_publish( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + command: String, + fields: Value, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + post_json( + &client, + port, + "/api/admin/publish", + &serde_json::json!({ "command": command, "fields": fields }), + ) + .await +} + +#[tauri::command] +pub async fn ms_welcome_whisper( + app: tauri::AppHandle, + registry: tauri::State<'_, TunnelRegistry>, + tunnel_id: String, + recipient_player_id: String, + source_player_id: String, + message: String, +) -> Result { + let port = tunnel_local_port(®istry, &tunnel_id)?; + let client = ensure_client(&app); + post_json( + &client, + port, + "/api/admin/welcome-whisper", + &serde_json::json!({ + "recipientPlayerId": recipient_player_id, + "sourcePlayerId": source_player_id, + "message": message, + }), + ) + .await +} + +fn search_path(base: &str, q: Option<&str>, limit: Option) -> String { + let mut out = base.to_string(); + let mut sep = '?'; + if let Some(qq) = q { + out.push(sep); + out.push_str(&format!("q={}", urlencoding(qq))); + sep = '&'; + } + if let Some(l) = limit { + out.push(sep); + out.push_str(&format!("limit={l}")); + } + out +} + +fn urlencoding(input: &str) -> String { + let mut out = String::with_capacity(input.len()); + for c in input.chars() { + match c { + 'A'..='Z' | 'a'..='z' | '0'..='9' | '-' | '_' | '.' | '~' => out.push(c), + _ => { + let mut buf = [0u8; 4]; + for byte in c.encode_utf8(&mut buf).bytes() { + out.push_str(&format!("%{:02X}", byte)); + } + } + } + } + out +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/management_service.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/management_service.rs new file mode 100644 index 0000000..fe95b81 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/management_service.rs @@ -0,0 +1,670 @@ +use std::path::PathBuf; + +use base64::Engine as _; +use dune_manager_core::orchestration::{RemoteCommandRunner, RusshRunner, RusshTarget}; +use serde::{Deserialize, Serialize}; +use tauri::{Emitter, Manager}; + +use crate::commands::shared::{command_error_message, sh_single_quoted}; + +const REMOTE_BINARY_PATH: &str = "/opt/dune-server-service/dune-server-service"; +const REMOTE_SYSTEMD_UNIT_PATH: &str = "/etc/systemd/system/dune-server-service.service"; +const REMOTE_OPENRC_PATH: &str = "/etc/init.d/dune-server-service"; + +const BUNDLED_VERSION: &str = env!("DUNE_SERVER_SERVICE_VERSION"); + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct ManagementInstallRequest { + pub host: String, + pub user: String, + pub key_path: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, + /// Optional command-auth token. If None, install only refreshes the binary. + pub command_auth_token: Option, +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct ManagementConnRequest { + pub host: String, + pub user: String, + pub key_path: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct ManagementInstallResult { + pub installed: bool, + pub started: bool, + pub init_system: String, + pub installed_version: Option, + pub message: String, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct ManagementServiceStatus { + pub installed: bool, + pub active: bool, + pub init_system: String, + pub installed_version: Option, + pub bundled_version: String, + pub journal_tail: String, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct InstallProgressEvent { + pub step: String, + pub status: String, + pub message: Option, +} + +fn default_ssh_port() -> u16 { + 22 +} + +#[derive(Debug, Clone)] +struct ServiceAccount { + user: String, + group: String, + home: String, +} + +fn target_from_conn(req: &ManagementConnRequest) -> Result { + let mut target = RusshTarget::new( + PathBuf::from( + req.key_path + .as_deref() + .unwrap_or_default() + .trim() + .to_string(), + ), + req.user.trim().to_string(), + req.host.trim().to_string(), + ); + if req.port != 0 { + target.port = req.port; + } + target.validate().map_err(|err| err.message)?; + Ok(target) +} + +fn target_from_install(req: &ManagementInstallRequest) -> Result { + let conn = ManagementConnRequest { + host: req.host.clone(), + user: req.user.clone(), + key_path: req.key_path.clone(), + port: req.port, + }; + target_from_conn(&conn) +} + +fn resolve_resource(app: &tauri::AppHandle, path: &str) -> Result { + let resource = app + .path() + .resolve(path, tauri::path::BaseDirectory::Resource) + .map_err(|err| format!("resolving bundled {path}: {err}"))?; + if !resource.exists() { + return Err(format!("bundled {path} missing at {}", resource.display())); + } + Ok(resource) +} + +#[tauri::command] +pub async fn install_management_service( + app: tauri::AppHandle, + request: ManagementInstallRequest, +) -> Result { + let binary_path = resolve_resource(&app, "binaries/dune-server-service")?; + let unit_path = resolve_resource(&app, "binaries/dune-server-service.service")?; + let openrc_path = resolve_resource(&app, "binaries/dune-server-service.openrc")?; + let target = target_from_install(&request)?; + let token = request.command_auth_token.clone(); + let app_handle = app.clone(); + + tauri::async_runtime::spawn_blocking(move || { + install_inner( + &app_handle, + &target, + &binary_path, + &unit_path, + &openrc_path, + token.as_deref(), + ) + }) + .await + .map_err(|err| format!("install worker failed: {err}"))? +} + +#[tauri::command] +pub fn management_service_bundled_version() -> String { + BUNDLED_VERSION.trim().to_string() +} + +#[tauri::command] +pub async fn uninstall_management_service(request: ManagementConnRequest) -> Result<(), String> { + let target = target_from_conn(&request)?; + tauri::async_runtime::spawn_blocking(move || uninstall_inner(&target)) + .await + .map_err(|err| format!("uninstall worker failed: {err}"))? +} + +#[tauri::command] +pub async fn restart_management_service(request: ManagementConnRequest) -> Result<(), String> { + let target = target_from_conn(&request)?; + tauri::async_runtime::spawn_blocking(move || { + let script = "set -eu\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + if command -v systemctl >/dev/null 2>&1; then\n \ + sudo systemctl restart dune-server-service.service\n\ + elif command -v rc-service >/dev/null 2>&1; then\n \ + sudo rc-service dune-server-service restart\n\ + else\n \ + echo \"no supported init system\" >&2\n \ + exit 1\n\ + fi\n\ + exit 0\n"; + let runner = RusshRunner::new(target.clone()); + runner + .run_script(script) + .map_err(command_error_message) + .map(|_| ()) + }) + .await + .map_err(|err| format!("restart worker failed: {err}"))? +} + +#[tauri::command] +pub async fn management_service_status( + request: ManagementConnRequest, +) -> Result { + let target = target_from_conn(&request)?; + tauri::async_runtime::spawn_blocking(move || status_inner(&target)) + .await + .map_err(|err| format!("status worker failed: {err}"))? +} + +fn install_inner( + app: &tauri::AppHandle, + target: &RusshTarget, + binary_path: &std::path::Path, + unit_path: &std::path::Path, + openrc_path: &std::path::Path, + token: Option<&str>, +) -> Result { + let runner = RusshRunner::new(target.clone()); + let account = discover_service_account(&runner, &target.user)?; + + emit_progress(app, "stop-old", "running", None); + let stop_script = "set +e\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + sudo systemctl disable --now server-management-service.service >/dev/null 2>&1 || true\n\ + sudo systemctl stop dune-server-service.service >/dev/null 2>&1 || true\n\ + sudo rc-service dune-server-service stop >/dev/null 2>&1 || true\n\ + exit 0\n"; + runner + .run_script(stop_script) + .map_err(|err| step_err(app, "stop-old", err))?; + emit_progress(app, "stop-old", "ok", None); + + emit_progress(app, "prepare-host", "running", None); + // Pre-create every directory the systemd unit lists under + // `ReadWritePaths=`. systemd sets up a mount namespace BEFORE the binary + // runs, and a missing path there is fatal (exit 226/NAMESPACE — see the + // "/root/.steam: No such file or directory" failure mode). The service's + // sqlite + OpenRC supervisor also need the state dir and log file owned + // by the service user up front; missing them produces the silent + // not-starting symptom (goofycoolguy / MadBuffoon / issues #5, #6). + let prepare_script = format!( + "set -eu\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + sudo install -d -m 0755 -o {user} -g {group} {home}/.dune\n\ + sudo install -d -m 0700 -o {user} -g {group} {state_dir}\n\ + sudo install -d -m 0755 -o {user} -g {group} {home}/.local\n\ + sudo install -d -m 0755 -o {user} -g {group} {home}/.local/bin\n\ + sudo install -d -m 0755 -o {user} -g {group} {home}/.steam\n\ + sudo install -d -m 0755 -o {user} -g {group} {home}/Steam\n\ + sudo touch /var/log/dune-server-service.log\n\ + sudo chown {user}:{group} /var/log/dune-server-service.log\n\ + sudo chmod 0644 /var/log/dune-server-service.log\n", + user = sh_single_quoted(&account.user), + group = sh_single_quoted(&account.group), + home = sh_single_quoted(&account.home), + state_dir = sh_single_quoted(&format!("{}/.dune/state", account.home)), + ); + runner + .run_script(&prepare_script) + .map_err(|err| step_err(app, "prepare-host", err))?; + emit_progress(app, "prepare-host", "ok", None); + + let binary_bytes = std::fs::read(binary_path) + .map_err(|err| format!("reading resource {}: {err}", binary_path.display()))?; + let binary_size = std::fs::metadata(binary_path) + .ok() + .map(|m| m.len()) + .unwrap_or(0); + let size_msg = if binary_size > 0 { + format!("{:.1} MB", binary_size as f64 / 1024.0 / 1024.0) + } else { + "unknown size".to_string() + }; + emit_progress( + app, + "upload-binary", + "running", + Some(format!( + "streaming {size_msg} from {} to {REMOTE_BINARY_PATH}", + binary_path.display() + )), + ); + let upload_script = format!( + "set -eu\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + sudo install -d -m 0755 /opt/dune-server-service\n\ + tmp=$(mktemp /tmp/dune-server-service.XXXXXX)\n\ + trap 'rm -f \"$tmp\"' EXIT\n\ + cat > \"$tmp\"\n\ + actual=$(wc -c < \"$tmp\" | tr -d '[:space:]')\n\ + if [ \"$actual\" != {expected_bytes} ]; then\n \ + echo \"upload byte-count mismatch: expected {expected_bytes}, got $actual\" >&2\n \ + exit 42\n\ + fi\n\ + sudo install -m 0755 -o root -g root \"$tmp\" {dest}\n\ + installed=$(sudo stat -c '%s bytes mode=%a owner=%U:%G' {dest})\n\ + echo \"remote install: $installed\"\n", + expected_bytes = binary_bytes.len(), + dest = sh_single_quoted(REMOTE_BINARY_PATH), + ); + let upload_stdout = runner + .run_with_stdin( + &format!("sh -c {}", sh_single_quoted(&upload_script)), + &binary_bytes, + ) + .map_err(|err| step_err(app, "upload-binary", err))?; + let upload_msg = if upload_stdout.trim().is_empty() { + size_msg + } else { + format!("{size_msg}; {}", upload_stdout.trim()) + }; + emit_progress(app, "upload-binary", "ok", Some(upload_msg)); + + if let Some(t) = token { + emit_progress(app, "write-token", "running", None); + let token_b64 = base64::engine::general_purpose::STANDARD.encode(t.as_bytes()); + let token_path = format!("{}/.dune/state/command-auth-token", account.home); + // Stage to a real temp file before `sudo install` instead of piping + // through `sudo install /dev/stdin ...`. On Ubuntu hosts with sudo + // `Defaults use_pty` (default on 24.04+), root-to-root sudo allocates + // a pty and the piped bytes never reach the child's fd 0, which + // surfaces as `install: No such file or directory` even though both + // /dev/stdin and the destination dir exist. The temp-file pattern + // sidesteps the pty entirely. + let token_script = format!( + "set -eu\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + sudo install -d -m 0700 -o {user} -g {group} {state_dir}\n\ + tmp=$(mktemp /tmp/dune-token.XXXXXX)\n\ + trap 'rm -f \"$tmp\"' EXIT\n\ + echo {b64} | base64 -d > \"$tmp\"\n\ + sudo install -m 0600 -o {user} -g {group} \"$tmp\" {dest}\n", + user = sh_single_quoted(&account.user), + group = sh_single_quoted(&account.group), + state_dir = sh_single_quoted(&format!("{}/.dune/state", account.home)), + b64 = sh_single_quoted(&token_b64), + dest = sh_single_quoted(&token_path), + ); + runner + .run_script(&token_script) + .map_err(|err| step_err(app, "write-token", err))?; + emit_progress(app, "write-token", "ok", None); + } else { + emit_progress( + app, + "write-token", + "ok", + Some("skipped (no token)".to_string()), + ); + } + + emit_progress(app, "install-init", "running", None); + let unit_b64 = base64::engine::general_purpose::STANDARD + .encode(render_systemd_unit(unit_path, &account)?.as_bytes()); + let openrc_b64 = base64::engine::general_purpose::STANDARD + .encode(render_openrc_unit(openrc_path, &account)?.as_bytes()); + // Stage unit content + drop-in to real temp files before `sudo install`. + // The previous `echo b64 | base64 -d | sudo install /dev/stdin ...` shape + // breaks on hosts where sudoers has `Defaults use_pty` enabled (default + // on Ubuntu 24.04+): root-to-root sudo allocates a pty and the piped + // bytes never reach the child's fd 0, surfacing as + // `install: No such file or directory`. mktemp + sudo install + // sidesteps the pty entirely. + let init_script = format!( + "set -eu\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + tmp_unit=$(mktemp /tmp/dune-unit.XXXXXX)\n\ + tmp_dropin=$(mktemp /tmp/dune-dropin.XXXXXX)\n\ + tmp_openrc=$(mktemp /tmp/dune-openrc.XXXXXX)\n\ + trap 'rm -f \"$tmp_unit\" \"$tmp_dropin\" \"$tmp_openrc\"' EXIT\n\ + if command -v systemctl >/dev/null 2>&1; then\n \ + echo SYSTEMD\n \ + echo {unit_b64} | base64 -d > \"$tmp_unit\"\n \ + sudo install -m 0644 -o root -g root \"$tmp_unit\" {unit_dest}\n \ + sudo install -d -m 0755 /etc/systemd/system/dune-server-service.service.d\n \ + printf '%s\\n' '[Service]' 'NoNewPrivileges=false' 'MemoryDenyWriteExecute=false' > \"$tmp_dropin\"\n \ + sudo install -m 0644 -o root -g root \"$tmp_dropin\" /etc/systemd/system/dune-server-service.service.d/zz-dune-steamcmd-compat.conf\n \ + sudo systemctl daemon-reload\n \ + sudo systemctl reset-failed dune-server-service.service >/dev/null 2>&1 || true\n\ + elif command -v rc-service >/dev/null 2>&1; then\n \ + echo OPENRC\n \ + echo {openrc_b64} | base64 -d > \"$tmp_openrc\"\n \ + sudo install -m 0755 -o root -g root \"$tmp_openrc\" {openrc_dest}\n \ + sudo rc-update add dune-server-service default >/dev/null 2>&1 || true\n\ + else\n \ + echo \"no supported init system found (need systemd or openrc)\" >&2\n \ + exit 1\n\ + fi\n", + unit_b64 = sh_single_quoted(&unit_b64), + unit_dest = sh_single_quoted(REMOTE_SYSTEMD_UNIT_PATH), + openrc_b64 = sh_single_quoted(&openrc_b64), + openrc_dest = sh_single_quoted(REMOTE_OPENRC_PATH), + ); + let init_stdout = runner + .run_script(&init_script) + .map_err(|err| step_err(app, "install-init", err))?; + let mut init_system = String::from("unknown"); + for line in init_stdout.lines() { + match line.trim() { + "SYSTEMD" => init_system = "systemd".to_string(), + "OPENRC" => init_system = "openrc".to_string(), + _ => {} + } + } + emit_progress(app, "install-init", "ok", Some(init_system.clone())); + + emit_progress(app, "start-service", "running", None); + let start_script = "set -eu\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + if command -v systemctl >/dev/null 2>&1; then\n \ + sudo systemctl enable --now dune-server-service.service\n\ + elif command -v rc-service >/dev/null 2>&1; then\n \ + sudo rc-service dune-server-service restart >/dev/null 2>&1 || sudo rc-service dune-server-service start\n\ + fi\n"; + runner + .run_script(start_script) + .map_err(|err| step_err(app, "start-service", err))?; + emit_progress(app, "start-service", "ok", None); + + emit_progress(app, "verify", "running", None); + // `STATE=...` line carries the canonical systemctl/openrc state. When the + // unit is anything other than active we also tail the journal so the UI + // surfaces *why* — empty parentheses helped nobody. + let verify_script = "set +e\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + if command -v systemctl >/dev/null 2>&1; then\n \ + sleep 1\n \ + state=$(sudo systemctl is-active dune-server-service.service 2>/dev/null | tr -d '\\r\\n')\n \ + [ -z \"$state\" ] && state=unknown\n \ + echo \"STATE=$state\"\n \ + if [ \"$state\" != active ]; then\n \ + echo '--- journalctl ---'\n \ + sudo journalctl -u dune-server-service.service -n 20 --no-pager 2>&1 | tail -n 20\n \ + fi\n\ + elif command -v rc-service >/dev/null 2>&1; then\n \ + sleep 1\n \ + if sudo rc-service dune-server-service status >/dev/null 2>&1; then echo STATE=active; else echo STATE=inactive; fi\n \ + if [ -f /var/log/dune-server-service.log ]; then\n \ + echo '--- supervisor log ---'\n \ + sudo tail -n 20 /var/log/dune-server-service.log 2>&1\n \ + fi\n\ + else\n \ + echo STATE=unknown\n\ + fi\n\ + /opt/dune-server-service/dune-server-service --version 2>/dev/null || true\n\ + exit 0\n"; + let verify_stdout = runner + .run_script(verify_script) + .map_err(|err| step_err(app, "verify", err))?; + let mut active_state = String::new(); + let mut installed_version: Option = None; + let mut diagnostic_lines: Vec = Vec::new(); + let mut collecting_diag = false; + for line in verify_stdout.lines() { + let trimmed = line.trim(); + if let Some(state) = trimmed.strip_prefix("STATE=") { + active_state = state.to_string(); + continue; + } + if trimmed.starts_with("--- ") && trimmed.ends_with(" ---") { + collecting_diag = true; + continue; + } + if trimmed.starts_with("dune-server-service ") { + installed_version = trimmed + .strip_prefix("dune-server-service ") + .map(|s| s.trim().to_string()); + continue; + } + if collecting_diag && !trimmed.is_empty() { + diagnostic_lines.push(trimmed.to_string()); + } + } + let started = active_state == "active"; + let verify_msg = match (started, &installed_version) { + (true, Some(v)) => Some(format!("active, version {v}")), + (true, None) => Some("active".to_string()), + (false, _) => { + let header = if active_state.is_empty() { + "not active".to_string() + } else { + format!("not active ({active_state})") + }; + if diagnostic_lines.is_empty() { + Some(header) + } else { + // Keep the tail short so the toast/log stays readable; full + // detail is still on the host via `journalctl -u ...`. + let tail: Vec = diagnostic_lines + .iter() + .rev() + .take(6) + .rev() + .cloned() + .collect(); + Some(format!("{header}\n{}", tail.join("\n"))) + } + } + }; + emit_progress( + app, + "verify", + if started { "ok" } else { "error" }, + verify_msg.clone(), + ); + + Ok(ManagementInstallResult { + installed: true, + started, + init_system: init_system.clone(), + installed_version, + message: format!("installed via {init_system}; active={active_state}"), + }) +} + +fn discover_service_account( + runner: &RusshRunner, + _registered_user: &str, +) -> Result { + // The Dune service ALWAYS runs as the vendor's `dune` user with home + // `/home/dune`, no matter which account the operator SSH'd in as. SSH + // login may be root / ubuntu / a custom sudoer; install steps escalate + // via `sudo install -o dune -g dune` and the systemd/openrc unit pins + // User=dune. We still call getent on the host to fail loudly if `dune` + // isn't provisioned yet (e.g. vendor setup wasn't run). + let script = "set -eu\n\ + user=dune\n\ + home=$(getent passwd \"$user\" | awk -F: '{print $6}')\n\ + group=$(id -gn \"$user\" 2>/dev/null || echo dune)\n\ + if [ -z \"$home\" ]; then\n \ + echo \"dune user not found on host — run the vendor setup first\" >&2\n \ + exit 1\n\ + fi\n\ + printf 'USER=%s\\nGROUP=%s\\nHOME=%s\\n' \"$user\" \"$group\" \"$home\"\n"; + let script = script.to_string(); + let stdout = runner.run_script(&script).map_err(command_error_message)?; + let mut account = ServiceAccount { + user: String::new(), + group: String::new(), + home: String::new(), + }; + for line in stdout.lines() { + if let Some(value) = line.strip_prefix("USER=") { + account.user = value.trim().to_string(); + } else if let Some(value) = line.strip_prefix("GROUP=") { + account.group = value.trim().to_string(); + } else if let Some(value) = line.strip_prefix("HOME=") { + account.home = value.trim().trim_end_matches('/').to_string(); + } + } + if account.user.is_empty() || account.group.is_empty() || account.home.is_empty() { + return Err(format!( + "could not resolve service account from remote output: {stdout}" + )); + } + Ok(account) +} + +fn render_systemd_unit(path: &std::path::Path, account: &ServiceAccount) -> Result { + let unit = std::fs::read_to_string(path) + .map_err(|err| format!("reading resource {}: {err}", path.display()))?; + let home = account.home.as_str(); + Ok(unit + .replace("User=dune", &format!("User={}", account.user)) + .replace("Group=dune", &format!("Group={}", account.group)) + .replace("/home/dune/.local/bin", &format!("{home}/.local/bin")) + .replace("/home/dune/.dune", &format!("{home}/.dune")) + .replace("/home/dune/.steam", &format!("{home}/.steam")) + .replace("/home/dune/Steam", &format!("{home}/Steam")) + .replace( + "Environment=\"DUNE_SERVICE_HOME=/home/dune\"", + &format!("Environment=\"DUNE_SERVICE_HOME={home}\""), + )) +} + +fn render_openrc_unit(path: &std::path::Path, account: &ServiceAccount) -> Result { + let unit = std::fs::read_to_string(path) + .map_err(|err| format!("reading resource {}: {err}", path.display()))?; + let home = account.home.as_str(); + Ok(unit + .replace( + "command_user=\"dune:dune\"", + &format!("command_user=\"{}:{}\"", account.user, account.group), + ) + .replace( + "--owner dune:dune", + &format!("--owner {}:{}", account.user, account.group), + ) + .replace("/home/dune/.dune", &format!("{home}/.dune")) + .replace( + "DUNE_SERVICE_HOME=\"${DUNE_SERVICE_HOME:-/home/dune}\"", + &format!("DUNE_SERVICE_HOME=\"${{DUNE_SERVICE_HOME:-{home}}}\""), + )) +} + +fn emit_progress(app: &tauri::AppHandle, step: &str, status: &str, message: Option) { + let payload = InstallProgressEvent { + step: step.to_string(), + status: status.to_string(), + message, + }; + let _ = app.emit("management-install-progress", payload); +} + +fn step_err( + app: &tauri::AppHandle, + step: &str, + err: dune_manager_core::models::CommandFailure, +) -> String { + let msg = command_error_message(err); + emit_progress(app, step, "error", Some(msg.clone())); + msg +} + +fn uninstall_inner(target: &RusshTarget) -> Result<(), String> { + let script = "set -eu\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + if command -v systemctl >/dev/null 2>&1; then\n \ + sudo systemctl disable --now dune-server-service.service >/dev/null 2>&1 || true\n \ + sudo rm -f /etc/systemd/system/dune-server-service.service\n \ + sudo systemctl daemon-reload\n\ + fi\n\ + if command -v rc-service >/dev/null 2>&1; then\n \ + sudo rc-service dune-server-service stop >/dev/null 2>&1 || true\n \ + sudo rc-update del dune-server-service default >/dev/null 2>&1 || true\n \ + sudo rm -f /etc/init.d/dune-server-service\n\ + fi\n\ + sudo rm -rf /opt/dune-server-service\n\ + exit 0\n"; + let runner = RusshRunner::new(target.clone()); + runner + .run_script(script) + .map_err(command_error_message) + .map(|_| ()) +} + +fn status_inner(target: &RusshTarget) -> Result { + let script = "set +e\n\ + export PATH=/sbin:/usr/sbin:/usr/local/sbin:$PATH\n\ + if [ -x /opt/dune-server-service/dune-server-service ]; then\n \ + echo INSTALLED=yes\n \ + /opt/dune-server-service/dune-server-service --version 2>/dev/null | head -n 1\n\ + else\n \ + echo INSTALLED=no\n\ + fi\n\ + if command -v systemctl >/dev/null 2>&1; then\n \ + echo INIT=systemd\n \ + sudo systemctl is-active dune-server-service.service\n\ + elif command -v rc-service >/dev/null 2>&1; then\n \ + echo INIT=openrc\n \ + sudo rc-service dune-server-service status >/dev/null 2>&1 && echo active || echo inactive\n\ + else\n \ + echo INIT=none\n\ + fi\n\ + exit 0\n"; + let runner = RusshRunner::new(target.clone()); + let stdout = runner.run_script(script).map_err(command_error_message)?; + let mut installed = false; + let mut active = false; + let mut init_system = String::from("unknown"); + let mut installed_version: Option = None; + for line in stdout.lines() { + let trimmed = line.trim(); + match trimmed { + "INSTALLED=yes" => installed = true, + "INSTALLED=no" => installed = false, + "INIT=systemd" => init_system = "systemd".to_string(), + "INIT=openrc" => init_system = "openrc".to_string(), + "INIT=none" => init_system = "none".to_string(), + "active" => active = true, + "inactive" => active = false, + other if other.starts_with("dune-server-service ") => { + installed_version = other + .strip_prefix("dune-server-service ") + .map(|s| s.trim().to_string()); + } + _ => {} + } + } + Ok(ManagementServiceStatus { + installed, + active, + init_system, + installed_version, + bundled_version: BUNDLED_VERSION.trim().to_string(), + journal_tail: String::new(), + }) +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/mod.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/mod.rs new file mode 100644 index 0000000..194c259 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/mod.rs @@ -0,0 +1,39 @@ +mod battlegroup; +mod component; +mod discovery; +mod logs; +mod management_api; +mod management_service; +mod preflight; +pub(crate) mod shared; +mod status; +mod status_data; +mod status_helpers; +mod status_naming; +mod tunnel; +mod tunnel_helpers; + +pub use battlegroup::{ + restart_remote_battlegroup, start_remote_battlegroup, stop_remote_battlegroup, + update_remote_battlegroup, +}; +pub use component::{remote_component_log_tail, restart_remote_component}; +pub use discovery::detect_remote_ubuntu_servers; +pub use logs::{get_logs_folder, record_operation_log}; +pub use management_api::{ + ms_cluster, ms_cron_preview, ms_dump_prune_execute, ms_dump_prune_preview, ms_get_config, + ms_health, ms_history, ms_list_commands, ms_list_logs, ms_list_runs, ms_list_timezones, + ms_player_location, ms_publish, ms_search_items, ms_search_journey_nodes, ms_search_players, + ms_search_skill_modules, ms_search_vehicles, ms_search_xp_event_tags, ms_set_config, + ms_trigger_run, ms_welcome_grant_retry, ms_welcome_grants, ms_welcome_whisper, +}; +pub use management_service::{ + install_management_service, management_service_bundled_version, management_service_status, + restart_management_service, uninstall_management_service, +}; +pub use preflight::check_remote_sudo; +pub use status::{remote_server_components, remote_server_status}; +pub use tunnel::{ + server_tunnel_status, start_custom_tunnel, start_server_tunnel, stop_all_tunnels, + stop_server_tunnel, +}; diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/preflight.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/preflight.rs new file mode 100644 index 0000000..8049574 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/preflight.rs @@ -0,0 +1,104 @@ +//! Pre-attach connectivity + sudo checks executed against a candidate host. + +use std::path::PathBuf; + +use dune_manager_core::orchestration::{RemoteCommandRunner, RusshRunner, RusshTarget}; +use serde::Serialize; + +#[derive(Debug, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct PreflightCheck { + /// SSH connection + key authentication succeeded. + pub ssh_ok: bool, + /// The SSH user can `sudo -n -u dune` without a password. + pub sudo_to_dune_ok: bool, + /// The `dune` user itself has passwordless sudo for arbitrary commands. + pub dune_nopasswd_ok: bool, + /// Whether the SSH login user IS `dune` (no impersonation needed). + pub is_dune_login: bool, + /// Raw stdout/stderr collected from the probe script — surfaced in the + /// UI when something fails so the operator can see exactly what + /// happened on the host. + pub raw_output: String, +} + +#[derive(Debug, serde::Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct PreflightRequest { + pub host: String, + pub user: String, + pub key_path: String, + #[serde(default)] + pub port: Option, +} + +/// Probes connectivity, SSH auth, and the various sudo capabilities we +/// rely on. The result is used to gate the attach flow with a clear error +/// banner when something is missing. +#[tauri::command] +pub async fn check_remote_sudo(request: PreflightRequest) -> Result { + let host = request.host.trim().to_string(); + let user = request.user.trim().to_string(); + let key_path = request.key_path.trim().to_string(); + let port = request.port; + if host.is_empty() || user.is_empty() || key_path.is_empty() { + return Err("Host, user, and SSH key path are required.".to_string()); + } + tauri::async_runtime::spawn_blocking(move || run_preflight(host, user, key_path, port)) + .await + .map_err(|err| format!("Preflight worker failed: {err}"))? +} + +fn run_preflight( + host: String, + user: String, + key_path: String, + port: Option, +) -> Result { + let mut target = RusshTarget::new(PathBuf::from(&key_path), user.clone(), host.clone()); + if let Some(p) = port { + target.port = p; + } + target.validate().map_err(|err| err.message)?; + let runner = RusshRunner::new(target); + let probe = r#"set +e +echo SSH_OK +if sudo -n -u dune true >/dev/null 2>&1; then echo SUDO_TO_DUNE_OK; else echo SUDO_TO_DUNE_FAILED; fi +if sudo -n -u dune sudo -n true >/dev/null 2>&1; then echo DUNE_NOPASSWD_OK; else echo DUNE_NOPASSWD_FAILED; fi +echo PREFLIGHT_DONE +"#; + let stdout = runner.run_script(probe).map_err(|err| { + // Connection / auth failures land here. Surface them to the UI so + // the operator can fix host/key before retrying. + if !err.stderr.trim().is_empty() { + format!("{}: {}", err.message, err.stderr.trim()) + } else { + err.message + } + })?; + let ssh_ok = stdout.contains("SSH_OK"); + let is_dune_login = user == "dune"; + // When the SSH login is already dune, we do not need a sudo-to-dune + // hop; treat it as ok regardless of the probe outcome. + let sudo_to_dune_ok = is_dune_login || stdout.contains("SUDO_TO_DUNE_OK"); + let dune_nopasswd_ok = if is_dune_login { + // `sudo -n -u dune sudo -n true` may be rejected when the outer + // sudo refuses self-targeting. Fall back to a direct `sudo -n true` + // check when the operator is already logged in as dune. Re-run a + // quick second probe. + let direct = r#"if sudo -n true >/dev/null 2>&1; then echo DUNE_NOPASSWD_OK; else echo DUNE_NOPASSWD_FAILED; fi"#; + runner + .run_script(direct) + .map(|out| out.contains("DUNE_NOPASSWD_OK")) + .unwrap_or(false) + } else { + stdout.contains("DUNE_NOPASSWD_OK") + }; + Ok(PreflightCheck { + ssh_ok, + sudo_to_dune_ok, + dune_nopasswd_ok, + is_dune_login, + raw_output: stdout, + }) +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/shared.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/shared.rs new file mode 100644 index 0000000..9f9e04f --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/shared.rs @@ -0,0 +1,47 @@ +use std::path::PathBuf; + +use dune_manager_core::models::CommandFailure; +use dune_manager_core::orchestration::{RusshRunner, RusshTarget}; + +pub fn remote_runner( + host: String, + user: String, + key_path: String, + port: Option, +) -> Result { + let mut target = RusshTarget::new(PathBuf::from(key_path), user, host); + if let Some(p) = port { + target.port = p; + } + target.validate().map_err(|err| err.message)?; + Ok(RusshRunner::new(target)) +} + +pub fn runner_for_remote_kind( + _server_type: Option<&str>, + host: String, + user: String, + key_path: Option, + port: Option, +) -> Result { + let key_path = key_path + .map(|value| value.trim().to_string()) + .filter(|value| !value.is_empty()) + .ok_or_else(|| "SSH private key is required for remote Ubuntu servers.".to_string())?; + remote_runner(host, user, key_path, port) +} + +pub fn command_error_message(err: CommandFailure) -> String { + let mut parts = vec![err.message]; + if !err.stderr.trim().is_empty() { + parts.push(err.stderr); + } + if !err.stdout.trim().is_empty() { + parts.push(err.stdout); + } + parts.join("\n") +} + +pub fn sh_single_quoted(value: &str) -> String { + format!("'{}'", value.replace('\'', "'\"'\"'")) +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status.rs new file mode 100644 index 0000000..317c9e9 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status.rs @@ -0,0 +1,40 @@ +use crate::commands::shared::{command_error_message, runner_for_remote_kind}; +use crate::commands::status_data::{read_remote_server_components, read_remote_server_status}; +use crate::dto::{RemoteServerActionRequest, RemoteServerComponent, RemoteServerStatus}; + +#[tauri::command] +pub async fn remote_server_status( + request: RemoteServerActionRequest, +) -> Result { + tauri::async_runtime::spawn_blocking(move || { + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host, + request.user, + request.key_path, + Some(request.port), + )?; + read_remote_server_status(&runner, &request.namespace, &request.battlegroup_name) + .map_err(command_error_message) + }) + .await + .map_err(|err| format!("Remote status worker failed: {err}"))? +} + +#[tauri::command] +pub async fn remote_server_components( + request: RemoteServerActionRequest, +) -> Result, String> { + tauri::async_runtime::spawn_blocking(move || { + let runner = runner_for_remote_kind( + request.server_type.as_deref(), + request.host, + request.user, + request.key_path, + Some(request.port), + )?; + read_remote_server_components(&runner, &request.namespace).map_err(command_error_message) + }) + .await + .map_err(|err| format!("Remote component diagnostics worker failed: {err}"))? +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_data.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_data.rs new file mode 100644 index 0000000..dd9fdba --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_data.rs @@ -0,0 +1,694 @@ +use dune_manager_core::errors::failure; +use dune_manager_core::models::CommandResult; +use dune_manager_core::orchestration::{RemoteCommandRunner, RusshRunner}; +use serde_json::Value; + +use crate::commands::shared::sh_single_quoted; +use crate::commands::status_helpers::{pod_component, server_resource_components}; +use crate::commands::status_naming::friendly_map_name; +use crate::dto::{ + RemoteBattlegroupServerStat, RemoteBattlegroupStatus, RemoteServerComponent, + RemoteServerPackageStatus, RemoteServerStatus, +}; + +pub fn read_remote_server_status( + runner: &RusshRunner, + namespace: &str, + battlegroup_name: &str, +) -> CommandResult { + // The vendor wrapper's `status` text output is the source of truth in + // older operator versions, but the format keeps shifting across Funcom + // releases (newer wrappers show the partial world name in "Status", + // "N/M" ratios under "Director", and semantic words like "Healthy" + // under "Uptime" — none of which match the older + // `Running/Running/Running/Running/1h2m` shape we used to parse). + // Read the BattleGroup CR's `status` object directly so we stay + // pinned to the stable Kubernetes schema instead of the rotating + // text rendering. + let bg = runner.run_json( + &format!( + "sudo kubectl get battlegroup -n {} {} -o json", + sh_single_quoted(namespace), + sh_single_quoted(battlegroup_name), + ), + "remote battlegroup", + )?; + // Per-partition live data (player count, gamePhase, ready) lives on a + // separate ServerStats CRD published by the Funcom operator — the same + // source `F:\Dune\Server\gt-server-status\gt_server_status.py` consumes. + // Failing to fetch this is non-fatal; the table just shows blank + // players where it can't be merged. + let stats = runner + .run_json( + &format!( + "sudo kubectl get serverstats -n {} -o json", + sh_single_quoted(namespace), + ), + "remote serverstats", + ) + .unwrap_or_else(|_| Value::Null); + let battlegroup = battlegroup_status_from_json_with_stats(&bg, &stats).ok_or_else(|| { + failure(format!( + "BattleGroup `{battlegroup_name}` returned no status object yet (likely still initialising)" + )) + })?; + let package = read_guest_package_status(runner, namespace, battlegroup_name)?; + Ok(RemoteServerStatus { + battlegroup, + package, + }) +} + +/// Maps a raw `kubectl get battlegroup ... -o json` payload into the UI's +/// `RemoteBattlegroupStatus` and merges per-partition +/// live data (players, gamePhase, ready) from a `kubectl get serverstats` +/// JSON payload. Pass `Value::Null` when no stats are available. +pub(crate) fn battlegroup_status_from_json_with_stats( + bg: &Value, + serverstats: &Value, +) -> Option { + bg.get("metadata")?.get("name")?.as_str()?; + let spec = bg.get("spec").cloned().unwrap_or(Value::Null); + let status = bg.get("status").cloned().unwrap_or(Value::Null); + + let stop = spec + .get("stop") + .and_then(Value::as_bool) + .or_else(|| status.get("stop").and_then(Value::as_bool)) + .unwrap_or(false); + + // Funcom's CR carries `status.startTimestamp` at the BG level (when the + // BG first scheduled) but not per-server. We render it on every row as a + // best-effort age — accurate when partitions all came up together, off + // by however long a partition has restarted independently. + let bg_age = status + .get("startTimestamp") + .and_then(Value::as_str) + .map(format_age_since_iso) + .unwrap_or_default(); + + let stats_by_partition = index_serverstats_by_partition(serverstats); + + let server_stats = status + .get("servers") + .and_then(Value::as_array) + .map(|servers| { + servers + .iter() + .map(|s| server_stat_from_json(s, &bg_age, &stats_by_partition)) + .collect() + }) + .unwrap_or_default(); + + // Database/director phases are nested in the live CR, not top-level + // fields. Fall back to top-level keys for older operator builds. + let database_phase = status + .get("database") + .and_then(|d| d.get("phase")) + .and_then(Value::as_str) + .map(str::to_string) + .unwrap_or_else(|| string_field(&status, "databasePhase")); + let director_phase = status + .get("utilities") + .and_then(|u| u.get("director")) + .and_then(|d| d.get("phase")) + .and_then(Value::as_str) + .map(str::to_string) + .unwrap_or_else(|| string_field(&status, "directorPhase")); + // Uptime: the CR doesn't expose a pre-formatted string anymore, so we + // compute it from `status.startTimestamp` (the same field we use for + // per-row age). Older operators that set a literal `uptime` string win. + let uptime_literal = string_field(&status, "uptime"); + let uptime = if uptime_literal.is_empty() { + bg_age.clone() + } else { + uptime_literal + }; + + Some(RemoteBattlegroupStatus { + stop, + phase: string_field(&status, "phase"), + database_phase, + server_group_phase: string_field(&status, "serverGroupPhase"), + director_phase, + uptime, + server_stats, + }) +} + +#[derive(Default, Clone)] +struct PartitionStats { + players: Option, +} + +/// Build a `partition_index -> PartitionStats` map from a `kubectl get +/// serverstats -n -o json` payload. The Funcom operator emits one +/// ServerStats CR per partition with `spec.area.partition` as the id and +/// `status.runtime.players` as the live count. Same source the +/// `gt_server_status.py` cron script consumes. +fn index_serverstats_by_partition(stats: &Value) -> std::collections::HashMap { + let mut out = std::collections::HashMap::new(); + let Some(items) = stats.get("items").and_then(Value::as_array) else { + return out; + }; + for item in items { + let partition = item + .get("spec") + .and_then(|s| s.get("area")) + .and_then(|a| a.get("partition")) + .and_then(Value::as_i64); + let Some(partition) = partition else { continue }; + let players = item + .get("status") + .and_then(|s| s.get("runtime")) + .and_then(|r| r.get("players")) + .and_then(Value::as_i64); + out.insert(partition, PartitionStats { players }); + } + out +} + +fn string_field(value: &Value, key: &str) -> String { + match value.get(key) { + Some(Value::String(s)) => s.clone(), + Some(Value::Number(n)) => n.to_string(), + Some(Value::Bool(b)) => b.to_string(), + _ => String::new(), + } +} + +fn server_stat_from_json( + server: &Value, + bg_age: &str, + stats_by_partition: &std::collections::HashMap, +) -> RemoteBattlegroupServerStat { + // The Funcom operator names this field `partitionMap` in the BattleGroup + // CR's `status.servers[]` — confirmed against backed-up live CR YAML. + // Older / alternate operators have used `map` or `name`, so we keep + // those as fallbacks. With no map at all `friendly_map_name` returns + // "Game Server" which is what we want to avoid here. + let raw_map = server + .get("partitionMap") + .and_then(Value::as_str) + .or_else(|| server.get("map").and_then(Value::as_str)) + .or_else(|| server.get("name").and_then(Value::as_str)) + .unwrap_or_default(); + let partition_index = server + .get("partitionIndex") + .and_then(Value::as_u64) + .or_else(|| server.get("ordinalIndex").and_then(Value::as_u64)); + let friendly = friendly_map_name(raw_map, raw_map); + let labelled = match partition_index { + Some(idx) => format!("{friendly} #{idx}"), + None => friendly, + }; + let ready_str = match server.get("ready") { + Some(Value::Bool(b)) => b.to_string(), + Some(Value::String(s)) => s.clone(), + Some(Value::Number(n)) => n.to_string(), + _ => String::new(), + }; + // The BG CR's status.servers[] entries don't carry a player count or + // age; we inherit the BG-level age and merge the per-partition player + // count from the matching ServerStats CR (keyed by partitionIndex). + let age = if let Some(start) = server.get("startTimestamp").and_then(Value::as_str) { + format_age_since_iso(start) + } else { + bg_age.to_string() + }; + let players = partition_index + .and_then(|idx| stats_by_partition.get(&(idx as i64))) + .and_then(|s| s.players) + .map(|n| n.to_string()) + .unwrap_or_default(); + RemoteBattlegroupServerStat { + map: labelled, + phase: string_field(server, "phase"), + ready: ready_str, + players, + age, + } +} + +/// Format an RFC 3339 timestamp like `"2026-05-22T01:27:53Z"` as a compact +/// elapsed-time string (`5d 3h`, `2h 17m`, `45m`, `12s`). Returns empty +/// string when parsing fails — the UI just shows an empty cell. +fn format_age_since_iso(iso_ts: &str) -> String { + let parsed = chrono::DateTime::parse_from_rfc3339(iso_ts.trim()); + let Ok(start) = parsed else { + return String::new(); + }; + let now = chrono::Utc::now(); + let diff = now.signed_duration_since(start.with_timezone(&chrono::Utc)); + let secs = diff.num_seconds().max(0); + if secs < 60 { + return format!("{secs}s"); + } + let minutes = secs / 60; + if minutes < 60 { + return format!("{minutes}m"); + } + let hours = minutes / 60; + let mins_rem = minutes % 60; + if hours < 24 { + return format!("{hours}h {mins_rem}m"); + } + let days = hours / 24; + let hours_rem = hours % 24; + format!("{days}d {hours_rem}h") +} + +fn read_guest_package_status( + runner: &RusshRunner, + namespace: &str, + battlegroup_name: &str, +) -> CommandResult { + let script = r#" +set -u +download=/home/dune/.dune/download +manifest="$download/steamapps/appmanifest_4754530.acf" +ns=__NAMESPACE__ +bg=__BATTLEGROUP__ +read_vdf_value() { + key="$1" + file="$2" + [ -f "$file" ] || return 0 + awk -F '"' -v wanted="$key" '$2 == wanted { print $4; exit }' "$file" 2>/dev/null || true +} +read_file() { + file="$1" + [ -f "$file" ] || return 0 + head -n 1 "$file" 2>/dev/null | tr -d '\r\n' +} +printf 'installedBuildId=%s\n' "$(read_vdf_value buildid "$manifest")" +printf 'battlegroupVersion=%s\n' "$(read_file "$download/images/battlegroup/version.txt")" +printf 'operatorVersion=%s\n' "$(read_file "$download/images/operators/version.txt")" +live_image=$(sudo kubectl get battlegroup "$bg" -n "$ns" -o jsonpath='{..image}' 2>/dev/null | tr ' ' '\n' | awk -F: '/self-hosting\/(igw-server|seabass-server):/ { print $NF; exit }' || true) +printf 'liveBattlegroupVersion=%s\n' "$live_image" +"# + .replace("__NAMESPACE__", &sh_single_quoted(namespace)) + .replace("__BATTLEGROUP__", &sh_single_quoted(battlegroup_name)); + let output = runner.run_script(&script)?; + let value = |key: &str| { + output.lines().find_map(|line| { + let (name, value) = line.split_once('=')?; + (name == key && !value.trim().is_empty()).then(|| value.trim().to_string()) + }) + }; + Ok(RemoteServerPackageStatus { + installed_build_id: value("installedBuildId"), + battlegroup_version: value("battlegroupVersion"), + live_battlegroup_version: value("liveBattlegroupVersion"), + operator_version: value("operatorVersion"), + }) +} + +pub fn read_remote_server_components( + runner: &RusshRunner, + namespace: &str, +) -> CommandResult> { + let pods = runner.run_json( + &format!( + "sudo kubectl get pods -n {} -o json", + sh_single_quoted(namespace) + ), + "remote server pods", + )?; + let resources = runner.run_json( + &format!( + "sudo kubectl get servergroups,servergateways,serversets -n {} -o json", + sh_single_quoted(namespace) + ), + "remote server resources", + )?; + + let mut components = vec![ + pod_component("Database", "database", &pods, |role, name| { + role.contains("database") && !name.contains("-util-") + }), + pod_component( + "Database utilities", + "database-utilities", + &pods, + |role, _| { + role.contains("database-utility") + || role.contains("database-monitor") + || role.contains("database-pghero") + }, + ), + pod_component("Message Queue", "message-queue", &pods, |role, name| { + role.contains("message-queue") || name.contains("-mq-") + }), + pod_component("Director", "director", &pods, |role, name| { + role.contains("battlegroup-director") || name.contains("-bgd-") + }), + pod_component("Gateway", "gateway", &pods, |role, name| { + role.contains("server-gateway") || name.contains("-sgw-") + }), + pod_component("Text Router", "text-router", &pods, |role, name| { + role.contains("text-router") || name.contains("-tr-") + }), + pod_component("File Browser", "file-browser", &pods, |role, name| { + role.contains("filebrowser") || name.contains("-fb-") + }), + ]; + components.extend(server_resource_components(&resources)); + Ok(components + .into_iter() + .filter(|component| component.state != "Not present") + .collect()) +} + +pub fn remote_records_from_battlegroups( + request: &crate::dto::RemoteConnectionRequest, + value: &Value, +) -> Vec { + value + .get("items") + .and_then(Value::as_array) + .into_iter() + .flatten() + .filter_map(|item| remote_record_from_battlegroup(request, item)) + .collect() +} + +fn remote_record_from_battlegroup( + request: &crate::dto::RemoteConnectionRequest, + item: &Value, +) -> Option { + let namespace = item + .get("metadata")? + .get("namespace")? + .as_str()? + .to_string(); + let battlegroup_name = item.get("metadata")?.get("name")?.as_str()?.to_string(); + let title = item + .get("spec") + .and_then(|spec| spec.get("title")) + .and_then(Value::as_str) + .unwrap_or(&battlegroup_name) + .to_string(); + let phase = item + .get("status") + .and_then(|status| status.get("phase")) + .and_then(Value::as_str) + .unwrap_or("Unknown") + .to_string(); + let server_type = request + .server_type + .as_deref() + .unwrap_or("ubuntu") + .trim() + .to_string(); + let user = request + .user + .as_deref() + .map(str::trim) + .unwrap_or_default() + .to_string(); + Some(crate::dto::RemoteServerRecord { + id: remote_record_id(&server_type, &request.host, request.key_path.as_deref()), + name: title, + host: request.host.clone(), + user, + key_path: request.key_path.clone().unwrap_or_default(), + port: request.port, + server_type, + namespace, + battlegroup_name: battlegroup_name.clone(), + world_unique_name: battlegroup_name, + phase, + }) +} + +fn remote_record_id(_server_type: &str, host: &str, key_path: Option<&str>) -> String { + format!( + "ubuntu:{}:{}", + host.trim().to_lowercase(), + key_path.unwrap_or_default().trim().to_lowercase() + ) +} + +#[cfg(test)] +mod tests { + use super::*; + use serde_json::json; + + fn bg(spec: Value, status: Value) -> Value { + json!({ + "metadata": {"name": "sh-test-bg", "namespace": "funcom-seabass-sh-test"}, + "spec": spec, + "status": status, + }) + } + + fn bg_status(bg: &Value) -> Option { + battlegroup_status_from_json_with_stats(bg, &Value::Null) + } + + #[test] + fn maps_reconciling_bg_with_null_director_phase() { + // Mirrors the user-reported payload: phase Reconciling, gateway + // Running, director not yet populated. Prior text-parse path was + // confusing the UI into greying the Director tunnel; under direct + // kubectl read the director_phase is just "" which the UI treats + // as "ready enough". + let value = bg( + json!({"stop": false}), + json!({ + "phase": "Reconciling", + "serverGroupPhase": "Running", + "directorPhase": Value::Null, + "stop": Value::Null, + }), + ); + let dto = bg_status(&value).expect("status maps"); + assert!(!dto.stop); + assert_eq!(dto.phase, "Reconciling"); + assert_eq!(dto.server_group_phase, "Running"); + assert_eq!(dto.director_phase, ""); + assert_eq!(dto.uptime, ""); + } + + #[test] + fn falls_back_to_status_stop_when_spec_missing() { + let value = bg(json!({}), json!({"phase": "Stopped", "stop": true})); + let dto = bg_status(&value).expect("status maps"); + assert!(dto.stop); + assert_eq!(dto.phase, "Stopped"); + } + + #[test] + fn server_stats_pulled_from_status_servers_array() { + let value = bg( + json!({"stop": false}), + json!({ + "phase": "Running", + "servers": [ + {"map": "Survival_1", "phase": "Running", "ready": true}, + {"name": "DeepDesert_1", "phase": "Stopped", "ready": false}, + ] + }), + ); + let dto = bg_status(&value).expect("status maps"); + assert_eq!(dto.server_stats.len(), 2); + assert_eq!( + dto.server_stats[0].map, + friendly_map_name("Survival_1", "Survival_1") + ); + assert_eq!(dto.server_stats[0].phase, "Running"); + assert_eq!(dto.server_stats[0].ready, "true"); + // Players empty when no ServerStats CR is supplied — that data lives + // on a separate CRD and is merged via `_with_stats`. + assert_eq!(dto.server_stats[0].players, ""); + assert_eq!( + dto.server_stats[1].map, + friendly_map_name("DeepDesert_1", "DeepDesert_1") + ); + assert_eq!(dto.server_stats[1].ready, "false"); + assert_eq!(dto.server_stats[1].age, ""); + } + + #[test] + fn server_stats_merge_player_count_from_serverstats_crd() { + // Mirrors the data shape gt_server_status.py reads: each ServerStats + // CR has spec.area.partition matching the BG's partitionIndex, and + // status.runtime.players is the live count. + let value = bg( + json!({"stop": false}), + json!({ + "phase": "Healthy", + "servers": [ + {"partitionMap": "Survival_1", "partitionIndex": 1, "phase": "Running", "ready": true}, + {"partitionMap": "Survival_1", "partitionIndex": 31, "phase": "Running", "ready": true}, + {"partitionMap": "Overmap", "partitionIndex": 2, "phase": "Running", "ready": true}, + ], + }), + ); + let stats = json!({ + "items": [ + {"spec": {"area": {"partition": 1, "map": "Survival_1"}}, "status": {"runtime": {"players": 7}}}, + {"spec": {"area": {"partition": 31, "map": "Survival_1"}}, "status": {"runtime": {"players": 0}}}, + {"spec": {"area": {"partition": 2, "map": "Overmap"}}, "status": {"runtime": {"players": 3}}}, + ], + }); + let dto = battlegroup_status_from_json_with_stats(&value, &stats).expect("status maps"); + assert_eq!(dto.server_stats[0].players, "7"); + assert_eq!(dto.server_stats[1].players, "0"); + assert_eq!(dto.server_stats[2].players, "3"); + } + + #[test] + fn server_stats_player_count_blank_when_partition_missing_from_stats() { + let value = bg( + json!({"stop": false}), + json!({ + "servers": [ + {"partitionMap": "Survival_1", "partitionIndex": 1, "phase": "Running", "ready": true}, + ], + }), + ); + let stats = json!({"items": []}); + let dto = battlegroup_status_from_json_with_stats(&value, &stats).expect("status maps"); + assert_eq!(dto.server_stats[0].players, ""); + } + + #[test] + fn server_stats_use_partition_map_and_index_from_real_cr() { + // Mirrors the actual Funcom operator status.servers[] shape captured + // from a live BattleGroup CR backup. Pre-fix the map column showed + // "Game Server" for every row because we were reading `map`/`name` + // instead of `partitionMap`. + let value = bg( + json!({"stop": false}), + json!({ + "phase": "Healthy", + "servers": [ + { + "partitionMap": "Survival_1", + "partitionIndex": 1, + "phase": "Running", + "ready": true, + }, + { + "partitionMap": "Survival_1", + "partitionIndex": 31, + "phase": "Running", + "ready": true, + }, + { + "partitionMap": "Overmap", + "partitionIndex": 2, + "phase": "Running", + "ready": true, + }, + ] + }), + ); + let dto = bg_status(&value).expect("status maps"); + assert_eq!(dto.server_stats.len(), 3); + assert_eq!(dto.server_stats[0].map, "Hagga Basin #1"); + assert_eq!(dto.server_stats[1].map, "Hagga Basin #31"); + assert_eq!(dto.server_stats[2].map, "Overmap #2"); + assert!(dto.server_stats.iter().all(|s| s.phase == "Running")); + assert!(dto.server_stats.iter().all(|s| s.ready == "true")); + } + + #[test] + fn returns_none_when_not_a_battlegroup_resource() { + let value = json!({"kind": "Pod", "spec": {}, "status": {}}); + assert!(bg_status(&value).is_none()); + } + + #[test] + fn bg_start_timestamp_propagates_to_every_server_row_when_per_server_missing() { + // status.startTimestamp from the live CR backup is one minute in the + // past for this test. + let one_min_ago = (chrono::Utc::now() - chrono::Duration::minutes(1)) + .to_rfc3339_opts(chrono::SecondsFormat::Secs, true); + let value = bg( + json!({"stop": false}), + json!({ + "phase": "Running", + "startTimestamp": one_min_ago, + "servers": [ + {"partitionMap": "Survival_1", "partitionIndex": 1, "phase": "Running", "ready": true}, + {"partitionMap": "Overmap", "partitionIndex": 2, "phase": "Running", "ready": true}, + ], + }), + ); + let dto = bg_status(&value).expect("status maps"); + // All rows pick up the same BG-level age. + assert_eq!(dto.server_stats.len(), 2); + for row in &dto.server_stats { + assert!( + row.age == "1m" || row.age == "60s", + "row age was {:?}", + row.age + ); + } + } + + #[test] + fn database_director_phases_pulled_from_nested_status() { + // Live CR shape: status.database.phase + status.utilities.director.phase, + // not top-level databasePhase/directorPhase. + let value = bg( + json!({"stop": false}), + json!({ + "phase": "Healthy", + "serverGroupPhase": "Running", + "database": {"phase": "Ready", "address": "1.2.3.4:15432"}, + "utilities": { + "director": {"phase": "Healthy", "address": "1.2.3.4:30393"}, + }, + }), + ); + let dto = bg_status(&value).expect("status maps"); + assert_eq!(dto.database_phase, "Ready"); + assert_eq!(dto.director_phase, "Healthy"); + } + + #[test] + fn uptime_derived_from_start_timestamp_when_no_literal() { + let one_hr_ago = + (chrono::Utc::now() - chrono::Duration::hours(1) - chrono::Duration::minutes(2)) + .to_rfc3339_opts(chrono::SecondsFormat::Secs, true); + let value = bg( + json!({"stop": false}), + json!({"phase": "Healthy", "startTimestamp": one_hr_ago}), + ); + let dto = bg_status(&value).expect("status maps"); + assert_eq!(dto.uptime, "1h 2m"); + } + + #[test] + fn uptime_prefers_literal_string_when_older_operator_set_it() { + let value = bg( + json!({"stop": false}), + json!({ + "phase": "Healthy", + "uptime": "1h2m", + "startTimestamp": "2026-05-22T01:27:53Z", + }), + ); + let dto = bg_status(&value).expect("status maps"); + assert_eq!(dto.uptime, "1h2m"); + } + + #[test] + fn format_age_since_iso_handles_common_shapes() { + assert_eq!(format_age_since_iso(""), ""); + assert_eq!(format_age_since_iso("not a timestamp"), ""); + let recent = (chrono::Utc::now() - chrono::Duration::seconds(30)) + .to_rfc3339_opts(chrono::SecondsFormat::Secs, true); + assert!(format_age_since_iso(&recent).ends_with('s')); + let hours = + (chrono::Utc::now() - chrono::Duration::hours(3) - chrono::Duration::minutes(15)) + .to_rfc3339_opts(chrono::SecondsFormat::Secs, true); + assert_eq!(format_age_since_iso(&hours), "3h 15m"); + let days = (chrono::Utc::now() - chrono::Duration::days(5) - chrono::Duration::hours(7)) + .to_rfc3339_opts(chrono::SecondsFormat::Secs, true); + assert_eq!(format_age_since_iso(&days), "5d 7h"); + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_helpers.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_helpers.rs new file mode 100644 index 0000000..989bb0f --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_helpers.rs @@ -0,0 +1,271 @@ +use serde_json::Value; + +use crate::commands::status_naming::{friendly_map_name, serverset_log_key}; +use crate::dto::RemoteServerComponent; + +pub fn pod_component( + label: &str, + log_key: &str, + pods: &Value, + matches: impl Fn(&str, &str) -> bool, +) -> RemoteServerComponent { + let mut total = 0usize; + let mut ready = 0usize; + let mut restarts = 0u64; + let mut reasons = Vec::new(); + let mut phases = Vec::new(); + for item in pods["items"].as_array().cloned().unwrap_or_default() { + let name = item["metadata"]["name"].as_str().unwrap_or_default(); + let role = item["metadata"]["labels"]["role"] + .as_str() + .unwrap_or_default(); + if !matches(role, name) { + continue; + } + total += 1; + let phase = item["status"]["phase"].as_str().unwrap_or_default(); + if !phase.is_empty() { + phases.push(phase.to_string()); + } + let statuses = item["status"]["containerStatuses"] + .as_array() + .cloned() + .unwrap_or_default(); + let pod_ready = !statuses.is_empty() + && statuses + .iter() + .all(|status| status["ready"].as_bool().unwrap_or(false)); + if pod_ready || phase == "Succeeded" { + ready += 1; + } + for status in statuses { + restarts += status["restartCount"].as_u64().unwrap_or_default(); + if let Some(reason) = status["state"]["waiting"]["reason"].as_str() { + reasons.push(reason.to_string()); + } + if let Some(reason) = status["state"]["terminated"]["reason"].as_str() { + if reason != "Completed" { + reasons.push(reason.to_string()); + } + } + } + } + + if total == 0 { + return component( + label, + log_key, + "system", + "Not present", + "gray", + "No matching runtime component was found.", + vec![], + ); + } + let details = compact_details(vec![ + format!("{ready}/{total} pods ready"), + if restarts > 0 { + format!("{restarts} container restarts") + } else { + String::new() + }, + if reasons.is_empty() { + String::new() + } else { + format!("Reason: {}", reasons.join(", ")) + }, + ]); + if ready == total && reasons.is_empty() { + component( + label, + log_key, + "system", + "Ready", + "green", + "All pods are ready.", + details, + ) + } else if reasons.iter().any(|reason| is_bad_reason(reason)) + || phases.iter().any(|phase| phase == "Failed") + { + component( + label, + log_key, + "system", + "Problem", + "red", + "One or more pods are failing.", + details, + ) + } else { + component( + label, + log_key, + "system", + "Starting", + "amber", + "Waiting for pods to become ready.", + details, + ) + } +} + +pub fn server_resource_components(resources: &Value) -> Vec { + let mut items = resources["items"].as_array().cloned().unwrap_or_default(); + items.sort_by(|left, right| { + left["metadata"]["name"] + .as_str() + .unwrap_or_default() + .cmp(right["metadata"]["name"].as_str().unwrap_or_default()) + }); + let mut output = Vec::new(); + for item in items { + let kind = item["kind"].as_str().unwrap_or_default(); + let name = item["metadata"]["name"].as_str().unwrap_or_default(); + match kind { + "ServerGroup" => output.push(server_group_component(&item)), + "ServerGateway" => output.push(resource_phase_component("Gateway Resource", &item)), + "ServerSet" => { + if should_show_serverset(&item) { + output.push(serverset_component(name, &item)); + } + } + _ => {} + } + } + output +} + +fn server_group_component(item: &Value) -> RemoteServerComponent { + let phase = item["status"]["phase"].as_str().unwrap_or("Unknown"); + phase_component( + "Server Group", + "server-group", + "system", + phase, + format!("Server Group reports {phase}."), + vec![], + ) +} + +fn resource_phase_component(label: &str, item: &Value) -> RemoteServerComponent { + let phase = item["status"]["phase"].as_str().unwrap_or("Unknown"); + phase_component( + label, + "gateway-resource", + "system", + phase, + format!("{label} reports {phase}."), + vec![], + ) +} + +fn serverset_component(name: &str, item: &Value) -> RemoteServerComponent { + let map = item["spec"]["map"].as_str().unwrap_or_default(); + let label = friendly_map_name(map, name); + let phase = item["status"]["phase"].as_str().unwrap_or("Unknown"); + let target = item["status"]["targetReplicas"] + .as_u64() + .unwrap_or_default(); + let ready = item["status"]["readyReplicas"].as_u64().unwrap_or_default(); + let completed = item["status"]["completedReplicas"] + .as_u64() + .unwrap_or_default(); + let pods = item["status"]["pods"] + .as_array() + .cloned() + .unwrap_or_default(); + let game_ready = pods + .iter() + .filter(|pod| pod["ready"].as_bool().unwrap_or(false)) + .count(); + let details = compact_details(vec![ + format!("{ready}/{target} Kubernetes-ready replicas"), + format!("{completed}/{target} completed game replicas"), + format!("{game_ready}/{target} game-ready servers"), + ]); + let summary = + if phase == "Initializing" && ready >= target && target > 0 && game_ready < target as usize + { + "Game process is running, but game readiness has not completed.".to_string() + } else { + format!("{label} reports {phase}.") + }; + phase_component( + &label, + &serverset_log_key(name, map), + "map", + phase, + summary, + details, + ) +} + +fn should_show_serverset(item: &Value) -> bool { + let phase = item["status"]["phase"].as_str().unwrap_or_default(); + let target = item["status"]["targetReplicas"] + .as_u64() + .unwrap_or_default(); + let map = item["spec"]["map"].as_str().unwrap_or_default(); + phase != "Stopped" || target > 0 || matches!(map, "Survival_1" | "Overmap" | "DeepDesert_1") +} + +fn phase_component( + label: &str, + log_key: &str, + category: &str, + phase: &str, + summary: String, + details: Vec, +) -> RemoteServerComponent { + let normalized = phase.to_ascii_lowercase(); + let (state, tone) = match normalized.as_str() { + "healthy" | "running" | "ready" | "available" => ("Ready", "green"), + "stopped" | "suspended" => ("Stopped", "gray"), + "initializing" | "reconciling" | "pending" | "starting" => ("Starting", "amber"), + "failed" | "error" | "degraded" => ("Problem", "red"), + _ => ("Unknown", "amber"), + }; + component(label, log_key, category, state, tone, summary, details) +} + +fn component( + name: &str, + log_key: &str, + category: &str, + state: &str, + tone: &str, + summary: impl Into, + details: Vec, +) -> RemoteServerComponent { + RemoteServerComponent { + name: name.to_string(), + log_key: log_key.to_string(), + category: category.to_string(), + state: state.to_string(), + tone: tone.to_string(), + summary: summary.into(), + details, + } +} + +fn compact_details(values: Vec) -> Vec { + values + .into_iter() + .filter(|value| !value.trim().is_empty()) + .collect() +} + +fn is_bad_reason(reason: &str) -> bool { + matches!( + reason, + "CrashLoopBackOff" + | "ImagePullBackOff" + | "ErrImagePull" + | "CreateContainerConfigError" + | "CreateContainerError" + | "RunContainerError" + | "OOMKilled" + | "Error" + ) +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_naming.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_naming.rs new file mode 100644 index 0000000..cb6a994 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/status_naming.rs @@ -0,0 +1,62 @@ +pub fn friendly_map_name(map: &str, fallback_name: &str) -> String { + let normalized = map.to_ascii_lowercase(); + if normalized == "survival_1" || fallback_name.contains("survival-1") { + return "Hagga Basin".to_string(); + } + if normalized == "overmap" || fallback_name.contains("overmap") { + return "Overmap".to_string(); + } + if normalized.contains("deepdesert") || fallback_name.contains("deepdesert") { + return "Deep Desert".to_string(); + } + if fallback_name.contains("sh-arrakeen") { + return "Social Hub: Arrakeen".to_string(); + } + if fallback_name.contains("sh-harkovillage") { + return "Social Hub: Harko Village".to_string(); + } + if !map.is_empty() { + return map.replace('_', " "); + } + "Game Server".to_string() +} + +pub fn serverset_log_key(name: &str, map: &str) -> String { + let combined = format!("{name} {map}").to_ascii_lowercase(); + if map.eq_ignore_ascii_case("Survival_1") || combined.contains("survival-1") { + return "map-survival-1".to_string(); + } + if map.eq_ignore_ascii_case("Overmap") || combined.contains("overmap") { + return "map-overmap".to_string(); + } + if combined.contains("deepdesert") || combined.contains("deep-desert") { + return "map-deepdesert".to_string(); + } + if combined.contains("sh-arrakeen") { + return "map-social-arrakeen".to_string(); + } + if combined.contains("sh-harkovillage") { + return "map-social-harkovillage".to_string(); + } + format!("map-{}", sanitize_component_key(map)) +} + +fn sanitize_component_key(value: &str) -> String { + let key = value + .chars() + .map(|character| { + if character.is_ascii_alphanumeric() { + character.to_ascii_lowercase() + } else { + '-' + } + }) + .collect::() + .trim_matches('-') + .to_string(); + if key.is_empty() { + "unknown".to_string() + } else { + key + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/tunnel.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/tunnel.rs new file mode 100644 index 0000000..e136aec --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/tunnel.rs @@ -0,0 +1,301 @@ +use std::io::{Read, Write}; +use std::net::TcpStream; +use std::path::PathBuf; +use std::time::Duration; + +use dune_manager_core::orchestration::{LocalForwarder, RusshTarget}; + +use crate::commands::tunnel_helpers::{ + discover_database_tunnel_port, discover_director_tunnel_port, discover_pg_hero_tunnel_port, + normalize_tunnel_service, tunnel_target, tunnel_url, +}; +use crate::dto::{ + CustomTunnelStartRequest, ServerTunnelStartRequest, ServerTunnelStatus, ServerTunnelStopRequest, +}; +use crate::state::{ManagedTunnel, TunnelRegistry}; + +const MANAGEMENT_API_PORT: u16 = 29187; +const LEGACY_MANAGEMENT_API_PORT: u16 = 8787; + +#[tauri::command] +pub async fn start_server_tunnel( + registry: tauri::State<'_, TunnelRegistry>, + request: ServerTunnelStartRequest, +) -> Result { + let registry = registry.inner().clone(); + tauri::async_runtime::spawn_blocking(move || start_server_tunnel_inner(®istry, request)) + .await + .map_err(|err| format!("Tunnel worker failed: {err}"))? +} + +#[tauri::command] +pub async fn stop_server_tunnel( + registry: tauri::State<'_, TunnelRegistry>, + request: ServerTunnelStopRequest, +) -> Result<(), String> { + let registry = registry.inner().clone(); + tauri::async_runtime::spawn_blocking(move || { + stop_server_tunnel_inner(®istry, &request.tunnel_id) + }) + .await + .map_err(|err| format!("Tunnel stop worker failed: {err}"))? +} + +#[tauri::command] +pub async fn server_tunnel_status( + registry: tauri::State<'_, TunnelRegistry>, + request: ServerTunnelStopRequest, +) -> Result, String> { + let registry = registry.inner().clone(); + tauri::async_runtime::spawn_blocking(move || { + existing_running_tunnel(®istry, request.tunnel_id.trim()) + }) + .await + .map_err(|err| format!("Tunnel status worker failed: {err}"))? +} + +#[tauri::command] +pub async fn stop_all_tunnels(registry: tauri::State<'_, TunnelRegistry>) -> Result<(), String> { + registry.stop_all(); + Ok(()) +} + +#[tauri::command] +pub async fn start_custom_tunnel( + registry: tauri::State<'_, TunnelRegistry>, + request: CustomTunnelStartRequest, +) -> Result { + let registry = registry.inner().clone(); + tauri::async_runtime::spawn_blocking(move || start_custom_tunnel_inner(®istry, request)) + .await + .map_err(|err| format!("Tunnel worker failed: {err}"))? +} + +fn start_custom_tunnel_inner( + registry: &TunnelRegistry, + request: CustomTunnelStartRequest, +) -> Result { + let tunnel_id = request.tunnel_id.trim(); + if tunnel_id.is_empty() { + return Err("Tunnel id is required.".to_string()); + } + if let Some(status) = existing_running_tunnel(registry, tunnel_id)? { + return Ok(status); + } + + let target = match request.server_kind.trim() { + "ubuntu" => { + let mut t = RusshTarget::new( + PathBuf::from(request.key_path.as_deref().unwrap_or_default().trim()), + request.user.trim().to_string(), + request.host.trim().to_string(), + ); + if request.port != 0 { + t.port = request.port; + } + t.validate().map_err(|err| err.message)?; + t + } + other => return Err(format!("Unsupported remote server kind: {other}")), + }; + + let forwarder = LocalForwarder::start( + &target, + request.local_port, + "127.0.0.1", + request.remote_port, + ) + .map_err(|err| err.message)?; + let local_port = forwarder.local_port(); + + let url = match request.protocol.trim() { + "https" => format!("https://127.0.0.1:{local_port}/"), + "postgresql" => format!("postgresql://127.0.0.1:{local_port}/"), + _ => format!("http://127.0.0.1:{local_port}/"), + }; + + let status = ServerTunnelStatus { + tunnel_id: tunnel_id.to_string(), + service: "custom".to_string(), + local_port, + remote_port: request.remote_port, + url, + }; + let mut tunnels = registry + .tunnels + .lock() + .map_err(|_| "Tunnel registry is unavailable.".to_string())?; + if let Some(existing) = tunnels.remove(tunnel_id) { + existing.forwarder.stop(); + } + tunnels.insert( + tunnel_id.to_string(), + ManagedTunnel { + forwarder, + status: status.clone(), + }, + ); + Ok(status) +} + +fn start_server_tunnel_inner( + registry: &TunnelRegistry, + request: ServerTunnelStartRequest, +) -> Result { + let tunnel_id = request.tunnel_id.trim(); + if tunnel_id.is_empty() { + return Err("Tunnel id is required.".to_string()); + } + if let Some(status) = existing_running_tunnel(registry, tunnel_id)? { + return Ok(status); + } + + let target = tunnel_target(&request)?; + let service = normalize_tunnel_service(&request.service)?; + let remote_port = match service.as_str() { + "director" => discover_director_tunnel_port(&target, &request.namespace)?, + "fileBrowser" => 18888, + "database" => discover_database_tunnel_port(&target, &request.namespace)?, + "pgHero" => discover_pg_hero_tunnel_port(&target, &request.namespace)?, + "managementApi" => MANAGEMENT_API_PORT, + _ => unreachable!(), + }; + + if service == "managementApi" { + return start_management_api_tunnel(registry, tunnel_id, &target, &service); + } + + let forwarder = + LocalForwarder::start(&target, 0, "127.0.0.1", remote_port).map_err(|err| err.message)?; + let local_port = forwarder.local_port(); + + let status = ServerTunnelStatus { + tunnel_id: tunnel_id.to_string(), + url: tunnel_url(&service, local_port), + service, + local_port, + remote_port, + }; + let mut tunnels = registry + .tunnels + .lock() + .map_err(|_| "Tunnel registry is unavailable.".to_string())?; + if let Some(existing) = tunnels.remove(tunnel_id) { + existing.forwarder.stop(); + } + tunnels.insert( + tunnel_id.to_string(), + ManagedTunnel { + forwarder, + status: status.clone(), + }, + ); + Ok(status) +} + +fn start_management_api_tunnel( + registry: &TunnelRegistry, + tunnel_id: &str, + target: &RusshTarget, + service: &str, +) -> Result { + let mut last_error = String::new(); + for remote_port in [MANAGEMENT_API_PORT, LEGACY_MANAGEMENT_API_PORT] { + let forwarder = LocalForwarder::start(target, 0, "127.0.0.1", remote_port) + .map_err(|err| err.message)?; + let local_port = forwarder.local_port(); + match probe_management_api(local_port) { + Ok(()) => { + let status = ServerTunnelStatus { + tunnel_id: tunnel_id.to_string(), + url: tunnel_url(service, local_port), + service: service.to_string(), + local_port, + remote_port, + }; + let mut tunnels = registry + .tunnels + .lock() + .map_err(|_| "Tunnel registry is unavailable.".to_string())?; + if let Some(existing) = tunnels.remove(tunnel_id) { + existing.forwarder.stop(); + } + tunnels.insert( + tunnel_id.to_string(), + ManagedTunnel { + forwarder, + status: status.clone(), + }, + ); + return Ok(status); + } + Err(err) => { + last_error = format!("127.0.0.1:{remote_port}: {err}"); + forwarder.stop(); + } + } + } + + Err(format!( + "management service did not answer on port {MANAGEMENT_API_PORT} or legacy port {LEGACY_MANAGEMENT_API_PORT}; last probe: {last_error}" + )) +} + +fn probe_management_api(local_port: u16) -> Result<(), String> { + let addr = format!("127.0.0.1:{local_port}"); + let timeout = Duration::from_millis(1500); + let socket_addr: std::net::SocketAddr = + addr.parse().map_err(|err| format!("bad addr: {err}"))?; + let mut stream = TcpStream::connect_timeout(&socket_addr, timeout) + .map_err(|err| format!("connect failed: {err}"))?; + stream.set_read_timeout(Some(timeout)).ok(); + stream.set_write_timeout(Some(timeout)).ok(); + stream + .write_all(b"GET /api/health HTTP/1.1\r\nHost: 127.0.0.1\r\nConnection: close\r\n\r\n") + .map_err(|err| format!("write failed: {err}"))?; + let mut buf = [0u8; 256]; + let n = stream + .read(&mut buf) + .map_err(|err| format!("read failed: {err}"))?; + if n == 0 { + return Err("remote closed without an HTTP response".to_string()); + } + let head = String::from_utf8_lossy(&buf[..n]); + if head.starts_with("HTTP/1.1 200") || head.starts_with("HTTP/1.0 200") { + Ok(()) + } else { + Err(format!("unexpected health response: {}", head.trim())) + } +} + +fn stop_server_tunnel_inner(registry: &TunnelRegistry, tunnel_id: &str) -> Result<(), String> { + let mut tunnels = registry + .tunnels + .lock() + .map_err(|_| "Tunnel registry is unavailable.".to_string())?; + if let Some(tunnel) = tunnels.remove(tunnel_id.trim()) { + tunnel.forwarder.stop(); + } + Ok(()) +} + +fn existing_running_tunnel( + registry: &TunnelRegistry, + tunnel_id: &str, +) -> Result, String> { + let mut tunnels = registry + .tunnels + .lock() + .map_err(|_| "Tunnel registry is unavailable.".to_string())?; + let Some(tunnel) = tunnels.get(tunnel_id) else { + return Ok(None); + }; + if tunnel.forwarder.is_finished() { + if let Some(stale) = tunnels.remove(tunnel_id) { + stale.forwarder.stop(); + } + Ok(None) + } else { + Ok(Some(tunnel.status.clone())) + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/commands/tunnel_helpers.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/tunnel_helpers.rs new file mode 100644 index 0000000..82c07e1 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/commands/tunnel_helpers.rs @@ -0,0 +1,156 @@ +use std::path::PathBuf; + +use dune_manager_core::orchestration::{RemoteCommandRunner, RusshRunner, RusshTarget}; + +use crate::commands::shared::{command_error_message, sh_single_quoted}; +use crate::dto::ServerTunnelStartRequest; + +pub fn tunnel_target(request: &ServerTunnelStartRequest) -> Result { + match request.server_kind.trim() { + "ubuntu" => { + let mut target = RusshTarget::new( + PathBuf::from( + request + .key_path + .as_deref() + .unwrap_or_default() + .trim() + .to_string(), + ), + request.user.trim().to_string(), + request.host.trim().to_string(), + ); + if request.port != 0 { + target.port = request.port; + } + target.validate().map_err(|err| err.message)?; + Ok(target) + } + other => Err(format!("Unsupported remote server kind: {other}")), + } +} + +pub fn normalize_tunnel_service(service: &str) -> Result { + match service.trim() { + "director" => Ok("director".to_string()), + "fileBrowser" => Ok("fileBrowser".to_string()), + "database" => Ok("database".to_string()), + "pgHero" => Ok("pgHero".to_string()), + "managementApi" => Ok("managementApi".to_string()), + other => Err(format!("Unsupported tunnel service: {other}")), + } +} + +pub fn tunnel_url(service: &str, local_port: u16) -> String { + match service { + "database" => format!("postgresql://127.0.0.1:{local_port}/dune"), + "managementApi" => format!("http://127.0.0.1:{local_port}/api"), + _ => format!("http://127.0.0.1:{local_port}/"), + } +} + +pub fn discover_director_tunnel_port(target: &RusshTarget, namespace: &str) -> Result { + let namespace = namespace.trim(); + if namespace.is_empty() { + return Err( + "BattleGroup namespace is required before starting the Director tunnel.".to_string(), + ); + } + let runner = RusshRunner::new(target.clone()); + let value = runner + .run_json( + &format!( + "sudo kubectl get svc -n {} -o json", + sh_single_quoted(namespace) + ), + "director service list", + ) + .map_err(command_error_message)?; + for service in value["items"].as_array().cloned().unwrap_or_default() { + for port in service["spec"]["ports"] + .as_array() + .cloned() + .unwrap_or_default() + { + if port["port"].as_u64() == Some(11717) { + if let Some(node_port) = port["nodePort"] + .as_u64() + .and_then(|value| u16::try_from(value).ok()) + { + return Ok(node_port); + } + } + } + } + Err("Director service is not currently exposed in Kubernetes.".to_string()) +} + +pub fn discover_database_tunnel_port(target: &RusshTarget, namespace: &str) -> Result { + const DEFAULT_DATABASE_PORT: u16 = dune_manager_core::database::DEFAULT_DUNE_DATABASE_PORT; + + let namespace = namespace.trim(); + if namespace.is_empty() { + return Err( + "BattleGroup namespace is required before starting the database tunnel.".to_string(), + ); + } + let runner = RusshRunner::new(target.clone()); + let value = runner + .run_json( + &format!( + "sudo kubectl get databasedeployments -n {} -o json", + sh_single_quoted(namespace) + ), + "database deployment list", + ) + .map_err(command_error_message)?; + for deployment in value["items"].as_array().cloned().unwrap_or_default() { + if let Some(port) = deployment["spec"]["port"] + .as_u64() + .and_then(|value| u16::try_from(value).ok()) + { + return Ok(port); + } + } + Ok(DEFAULT_DATABASE_PORT) +} + +pub fn discover_pg_hero_tunnel_port(target: &RusshTarget, namespace: &str) -> Result { + const DEFAULT_PG_HERO_PORT: u16 = 21111; + + let namespace = namespace.trim(); + if namespace.is_empty() { + return Err( + "BattleGroup namespace is required before starting the PgHero tunnel.".to_string(), + ); + } + let runner = RusshRunner::new(target.clone()); + let value = runner + .run_json( + &format!( + "sudo kubectl get pods -n {} -l role=igw-database-pghero -o json", + sh_single_quoted(namespace) + ), + "PgHero pod list", + ) + .map_err(command_error_message)?; + for pod in value["items"].as_array().cloned().unwrap_or_default() { + for container in pod["spec"]["containers"] + .as_array() + .cloned() + .unwrap_or_default() + { + for env in container["env"].as_array().cloned().unwrap_or_default() { + if env["name"].as_str() == Some("PORT") { + if let Some(port) = env["value"] + .as_str() + .and_then(|value| value.parse::().ok()) + { + return Ok(port); + } + } + } + } + } + Ok(DEFAULT_PG_HERO_PORT) +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/dto.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/dto.rs new file mode 100644 index 0000000..eb260cd --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/dto.rs @@ -0,0 +1,186 @@ +use serde::{Deserialize, Serialize}; + +fn default_ssh_port() -> u16 { + 22 +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteConnectionRequest { + pub host: String, + pub key_path: Option, + pub server_type: Option, + pub user: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteServerActionRequest { + pub server_type: Option, + pub host: String, + pub user: String, + pub key_path: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, + pub namespace: String, + pub battlegroup_name: String, +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct ServerTunnelStartRequest { + pub tunnel_id: String, + pub server_kind: String, + pub service: String, + pub host: String, + pub user: String, + pub key_path: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, + pub namespace: String, +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct ServerTunnelStopRequest { + pub tunnel_id: String, +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct CustomTunnelStartRequest { + pub tunnel_id: String, + pub server_kind: String, + pub host: String, + pub user: String, + pub key_path: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, + pub protocol: String, + pub remote_port: u16, + pub local_port: u16, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct ServerTunnelStatus { + pub tunnel_id: String, + pub service: String, + pub local_port: u16, + pub remote_port: u16, + pub url: String, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteBattlegroupStatus { + pub stop: bool, + pub phase: String, + #[serde(default)] + pub database_phase: String, + /// Wrapper's `Gateway` column. Kept under the old name for UI compatibility. + pub server_group_phase: String, + pub director_phase: String, + #[serde(default)] + pub uptime: String, + #[serde(default)] + pub server_stats: Vec, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteBattlegroupServerStat { + pub map: String, + pub phase: String, + pub ready: String, + pub players: String, + pub age: String, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteServerStatus { + pub battlegroup: RemoteBattlegroupStatus, + pub package: RemoteServerPackageStatus, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteServerPackageStatus { + pub installed_build_id: Option, + pub battlegroup_version: Option, + pub live_battlegroup_version: Option, + pub operator_version: Option, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteServerComponent { + pub name: String, + pub log_key: String, + pub category: String, + pub state: String, + pub tone: String, + pub summary: String, + pub details: Vec, +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteComponentLogRequest { + pub server_type: Option, + pub host: String, + pub user: String, + pub key_path: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, + pub namespace: String, + pub component: String, + pub tail: u32, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteComponentLogResult { + pub component: String, + pub output: String, +} + +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteComponentRestartRequest { + pub server_type: Option, + pub host: String, + pub user: String, + pub key_path: Option, + #[serde(default = "default_ssh_port")] + pub port: u16, + pub namespace: String, + pub component: String, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteComponentRestartResult { + pub component: String, + pub output: String, +} + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct RemoteServerRecord { + #[serde(rename = "type")] + pub server_type: String, + pub id: String, + pub name: String, + pub host: String, + pub user: String, + pub key_path: String, + pub port: u16, + pub namespace: String, + pub battlegroup_name: String, + pub world_unique_name: String, + pub phase: String, +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/lib.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/lib.rs new file mode 100644 index 0000000..e4fce85 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/lib.rs @@ -0,0 +1,112 @@ +mod commands; +mod dto; +mod log_file; +mod logging; +mod state; + +use std::sync::Arc; + +use tauri::Manager; + +use crate::log_file::LogFile; + +use crate::commands::{ + check_remote_sudo, detect_remote_ubuntu_servers, get_logs_folder, install_management_service, + management_service_bundled_version, management_service_status, ms_cluster, ms_cron_preview, + ms_dump_prune_execute, ms_dump_prune_preview, ms_get_config, ms_health, ms_history, + ms_list_commands, ms_list_logs, ms_list_runs, ms_list_timezones, ms_player_location, + ms_publish, ms_search_items, ms_search_journey_nodes, ms_search_players, + ms_search_skill_modules, ms_search_vehicles, ms_search_xp_event_tags, ms_set_config, + ms_trigger_run, ms_welcome_grant_retry, ms_welcome_grants, ms_welcome_whisper, + record_operation_log, + remote_component_log_tail, remote_server_components, remote_server_status, + restart_management_service, restart_remote_battlegroup, restart_remote_component, + server_tunnel_status, start_custom_tunnel, start_remote_battlegroup, start_server_tunnel, + stop_all_tunnels, stop_remote_battlegroup, stop_server_tunnel, uninstall_management_service, + update_remote_battlegroup, +}; +use crate::state::TunnelRegistry; + +#[cfg_attr(mobile, tauri::mobile_entry_point)] +pub fn run() { + // WebKitGTK 4.1 (Fedora 40+, WebKit 2.44+) aborts under GNOME Wayland + // with "Error 71 dispatching to Wayland display" when the DMABuf + // renderer is active. Disable it unless the user opted in explicitly. + #[cfg(target_os = "linux")] + if std::env::var_os("WEBKIT_DISABLE_DMABUF_RENDERER").is_none() { + std::env::set_var("WEBKIT_DISABLE_DMABUF_RENDERER", "1"); + } + + tauri::Builder::default() + .manage(TunnelRegistry::default()) + .plugin(tauri_plugin_dialog::init()) + .plugin(tauri_plugin_process::init()) + .plugin(tauri_plugin_shell::init()) + .plugin(tauri_plugin_updater::Builder::new().build()) + .setup(|app| { + match LogFile::new(&app.handle()) { + Ok(file) => { + app.manage(Arc::new(file)); + } + Err(err) => { + eprintln!("Failed to initialize operation log file: {err}"); + } + } + Ok(()) + }) + .invoke_handler(tauri::generate_handler![ + remote_server_status, + remote_server_components, + start_server_tunnel, + start_custom_tunnel, + stop_server_tunnel, + server_tunnel_status, + stop_all_tunnels, + remote_component_log_tail, + restart_remote_component, + start_remote_battlegroup, + stop_remote_battlegroup, + restart_remote_battlegroup, + update_remote_battlegroup, + detect_remote_ubuntu_servers, + check_remote_sudo, + record_operation_log, + get_logs_folder, + install_management_service, + uninstall_management_service, + management_service_status, + management_service_bundled_version, + restart_management_service, + ms_get_config, + ms_set_config, + ms_list_timezones, + ms_cron_preview, + ms_dump_prune_preview, + ms_dump_prune_execute, + ms_player_location, + ms_health, + ms_list_runs, + ms_list_logs, + ms_trigger_run, + ms_list_commands, + ms_search_items, + ms_search_vehicles, + ms_search_players, + ms_search_skill_modules, + ms_search_journey_nodes, + ms_search_xp_event_tags, + ms_cluster, + ms_history, + ms_welcome_grants, + ms_welcome_grant_retry, + ms_welcome_whisper, + ms_publish, + ]) + .on_window_event(|window, event| { + if matches!(event, tauri::WindowEvent::CloseRequested { .. }) { + window.state::().stop_all(); + } + }) + .run(tauri::generate_context!()) + .expect("failed to run Tauri application"); +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/log_file.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/log_file.rs new file mode 100644 index 0000000..17df22c --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/log_file.rs @@ -0,0 +1,131 @@ +//! Append-only operation log file with simple size-based rotation. + +use std::fs::{self, File, OpenOptions}; +use std::io::Write; +use std::path::{Path, PathBuf}; +use std::sync::Mutex; +use std::time::{SystemTime, UNIX_EPOCH}; + +use serde::Serialize; +use tauri::{AppHandle, Manager}; + +const MAX_LOG_BYTES: u64 = 10 * 1024 * 1024; +const LOG_FILE_NAME: &str = "operation.log"; +const ROLLED_FILE_NAME: &str = "operation.log.1"; + +#[derive(Debug, Serialize)] +struct LogLine<'a> { + ts: String, + level: &'a str, + scope: &'a str, + message: &'a str, +} + +/// JSON-line append-only sink for operation logs. +pub struct LogFile { + dir: PathBuf, + path: PathBuf, + file: Mutex, +} + +impl LogFile { + /// Resolves the app's local log directory, creates it if missing, and + /// opens `operation.log` for append. Errors out only if the directory + /// cannot be created or the file cannot be opened. + pub fn new(app: &AppHandle) -> std::io::Result { + let dir = app + .path() + .app_log_dir() + .map_err(|err| std::io::Error::new(std::io::ErrorKind::Other, err.to_string()))?; + fs::create_dir_all(&dir)?; + let path = dir.join(LOG_FILE_NAME); + let file = OpenOptions::new().create(true).append(true).open(&path)?; + Ok(Self { + dir, + path, + file: Mutex::new(file), + }) + } + + /// Returns the directory the log file lives in. + pub fn dir(&self) -> &Path { + &self.dir + } + + /// Appends a single JSON-line entry. Errors are swallowed by callers + /// because the live in-memory log view is the source of truth. + pub fn append(&self, level: &str, scope: &str, message: &str) -> std::io::Result<()> { + let line = LogLine { + ts: iso_timestamp(), + level, + scope, + message, + }; + let mut text = serde_json::to_string(&line) + .map_err(|err| std::io::Error::new(std::io::ErrorKind::Other, err.to_string()))?; + text.push('\n'); + let mut file = self + .file + .lock() + .map_err(|err| std::io::Error::new(std::io::ErrorKind::Other, err.to_string()))?; + file.write_all(text.as_bytes())?; + self.maybe_rotate_locked(&mut file)?; + Ok(()) + } + + fn maybe_rotate_locked(&self, file: &mut File) -> std::io::Result<()> { + let len = file.metadata()?.len(); + if len < MAX_LOG_BYTES { + return Ok(()); + } + // Drop the file handle before renaming on Windows. + drop(std::mem::replace( + file, + OpenOptions::new().read(true).open(&self.path)?, + )); + let rolled = self.dir.join(ROLLED_FILE_NAME); + let _ = fs::remove_file(&rolled); + fs::rename(&self.path, &rolled)?; + *file = OpenOptions::new() + .create(true) + .append(true) + .open(&self.path)?; + Ok(()) + } +} + +fn iso_timestamp() -> String { + let now = SystemTime::now() + .duration_since(UNIX_EPOCH) + .unwrap_or_default(); + let secs = now.as_secs(); + let millis = now.subsec_millis(); + // Minimal ISO-8601 UTC formatter without bringing in chrono. + let days_from_epoch = (secs / 86_400) as i64; + let (year, month, day) = civil_from_days(days_from_epoch); + let seconds_in_day = secs % 86_400; + let hour = (seconds_in_day / 3600) as u32; + let minute = ((seconds_in_day / 60) % 60) as u32; + let second = (seconds_in_day % 60) as u32; + format!("{year:04}-{month:02}-{day:02}T{hour:02}:{minute:02}:{second:02}.{millis:03}Z") +} + +/// Converts days-since-1970 to a (year, month, day) Gregorian triple. +/// Based on Howard Hinnant's `civil_from_days` algorithm. +fn civil_from_days(z: i64) -> (i64, u32, u32) { + let z = z + 719_468; + let era = if z >= 0 { z } else { z - 146_096 } / 146_097; + let doe = (z - era * 146_097) as u64; + let yoe = (doe - doe / 1_460 + doe / 36_524 - doe / 146_096) / 365; + let y = yoe as i64 + era * 400; + let doy = doe - (365 * yoe + yoe / 4 - yoe / 100); + let mp = (5 * doy + 2) / 153; + let d = (doy - (153 * mp + 2) / 5 + 1) as u32; + let m = if mp < 10 { + (mp + 3) as u32 + } else { + (mp - 9) as u32 + }; + let y = if m <= 2 { y + 1 } else { y }; + (y, m, d) +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/logging.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/logging.rs new file mode 100644 index 0000000..58da450 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/logging.rs @@ -0,0 +1,65 @@ +use std::sync::Arc; + +use dune_manager_core::orchestration::{OperationSink, OrchestrationEvent}; +use serde::Serialize; +use tauri::{AppHandle, Emitter, Manager}; + +use crate::log_file::LogFile; + +#[derive(Debug, Clone, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct OperationLogPayload { + pub level: &'static str, + pub scope: String, + pub message: String, + #[serde(skip_serializing_if = "Option::is_none")] + pub server_id: Option, +} + +pub struct TauriOperationSink { + pub app: AppHandle, + pub server_id: Option, +} + +impl TauriOperationSink { + pub fn new(app: AppHandle) -> Self { + Self { + app, + server_id: None, + } + } + + pub fn info(&self, scope: impl Into, message: impl Into) { + self.emit_level("info", scope, message); + } + + pub fn warn(&self, scope: impl Into, message: impl Into) { + self.emit_level("warn", scope, message); + } + + fn emit_level( + &self, + level: &'static str, + scope: impl Into, + message: impl Into, + ) { + let scope_text = scope.into(); + let message_text = message.into(); + let payload = OperationLogPayload { + level, + scope: scope_text.clone(), + message: message_text.clone(), + server_id: self.server_id.clone(), + }; + let _ = self.app.emit("operation-log", &payload); + if let Some(log_file) = self.app.try_state::>() { + let _ = log_file.append(level, &scope_text, &message_text); + } + } +} + +impl OperationSink for TauriOperationSink { + fn emit(&mut self, event: OrchestrationEvent) { + self.info(event.step_id, event.message); + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/main.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/main.rs new file mode 100644 index 0000000..1ca792c --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/main.rs @@ -0,0 +1,5 @@ +#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] + +fn main() { + dune_dedicated_server_manager_app_lib::run(); +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/src/state.rs b/docs/reference-repos/adainrivers/app/src-tauri/src/state.rs new file mode 100644 index 0000000..0bb1dbd --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/src/state.rs @@ -0,0 +1,29 @@ +use std::{ + collections::HashMap, + sync::{Arc, Mutex}, +}; + +use dune_manager_core::orchestration::LocalForwarder; + +use crate::dto::ServerTunnelStatus; + +#[derive(Default, Clone)] +pub struct TunnelRegistry { + pub tunnels: Arc>>, +} + +pub struct ManagedTunnel { + pub forwarder: LocalForwarder, + pub status: ServerTunnelStatus, +} + +impl TunnelRegistry { + pub fn stop_all(&self) { + let Ok(mut tunnels) = self.tunnels.lock() else { + return; + }; + for (_, tunnel) in tunnels.drain() { + tunnel.forwarder.stop(); + } + } +} diff --git a/docs/reference-repos/adainrivers/app/src-tauri/tauri.conf.json b/docs/reference-repos/adainrivers/app/src-tauri/tauri.conf.json new file mode 100644 index 0000000..90a022a --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src-tauri/tauri.conf.json @@ -0,0 +1,55 @@ +{ + "$schema": "https://schema.tauri.app/config/2", + "productName": "Dune Dedicated Server Manager", + "version": "0.3.16", + "identifier": "dev.dune.dedicated-server-manager", + "build": { + "beforeDevCommand": "npm run dev", + "devUrl": "http://127.0.0.1:1420", + "beforeBuildCommand": "npm run build", + "frontendDist": "../dist" + }, + "app": { + "windows": [ + { + "title": "Dune Dedicated Server Manager", + "width": 1280, + "height": 820, + "minWidth": 960, + "minHeight": 640, + "devtools": true + } + ], + "security": { + "csp": null + } + }, + "bundle": { + "active": true, + "targets": ["nsis"], + "createUpdaterArtifacts": false, + "resources": { + "binaries/dune-server-service": "binaries/dune-server-service", + "binaries/dune-server-service.service": "binaries/dune-server-service.service", + "binaries/dune-server-service.openrc": "binaries/dune-server-service.openrc" + }, + "icon": [ + "icons/32x32.png", + "icons/128x128.png", + "icons/128x128@2x.png", + "icons/icon.icns", + "icons/icon.ico" + ] + }, + "plugins": { + "updater": { + "pubkey": "dW50cnVzdGVkIGNvbW1lbnQ6IG1pbmlzaWduIHB1YmxpYyBrZXk6IEE2N0QwRkY3NzBBNTMxNTAKUldSUU1hVnc5dzk5cGg2b3ZVKzNpUGI0M3BNVHdzWk1mdzQwSzMrdm9HVllvQlg5bEdyM1BCU0UK", + "endpoints": [ + "https://github.com/adainrivers/dune-dedicated-server-manager/releases/latest/download/latest.json" + ], + "windows": { + "installMode": "passive" + } + } + } +} diff --git a/docs/reference-repos/adainrivers/app/src/App.tsx b/docs/reference-repos/adainrivers/app/src/App.tsx new file mode 100644 index 0000000..0c0b065 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/App.tsx @@ -0,0 +1,193 @@ +import { Box, Flex, Theme } from "@radix-ui/themes"; + +import AppErrorBoundary from "./components/AppErrorBoundary"; +import Header from "./components/layout/Header"; +import LogWindow from "./components/logs/LogWindow"; +import RemoteAttachDialog from "./components/dialogs/RemoteAttachDialog"; +import RemoveRemoteServerDialog from "./components/dialogs/RemoveRemoteServerDialog"; +import UpdateDialog from "./components/dialogs/UpdateDialog"; +import ServerDetailPage from "./components/servers/ServerDetailPage"; +import ServersListPage from "./components/servers/ServersListPage"; +import { useAppUpdates } from "./hooks/useAppUpdates"; +import { useComponentActions } from "./hooks/useComponentActions"; +import { useOperationLogs } from "./hooks/useOperationLogs"; +import { useRemoteServerStatus } from "./hooks/useRemoteServerStatus"; +import { useRemoteServers } from "./hooks/useRemoteServers"; +import { useServerTunnels } from "./hooks/useServerTunnels"; +import { useActivePage } from "./hooks/useActivePage"; +import { log } from "./utils/logging"; + +export function App() { + const { + logLevelFilter, + setLogLevelFilter, + logPanelCollapsed, + setLogPanelCollapsed, + scopeToActiveServer, + setScopeToActiveServer, + appendLogRow, + clearLogRows, + renderedLogRows, + } = useOperationLogs(); + + const remoteServersHook = useRemoteServers({ appendLogRow }); + + const tunnels = useServerTunnels({ appendLogRow }); + + const status = useRemoteServerStatus({ + appendLogRow, + setRemoteServers: remoteServersHook.setRemoteServers, + }); + + const componentActions = useComponentActions({ + appendLogRow, + detectRemoteServerDetails: status.detectRemoteServerDetails, + setRemoteServerComponents: status.setRemoteServerComponents, + setRemoteComponentLogs: status.setRemoteComponentLogs, + setRemoteComponentLogBusy: status.setRemoteComponentLogBusy, + setRemoteComponentRestartBusy: status.setRemoteComponentRestartBusy, + }); + + const updates = useAppUpdates({ appendLogRow }); + + remoteServersHook.bindRefreshRemoteServerStatus(status.refreshRemoteServerStatus); + remoteServersHook.bindRemoteServerBusy(status.remoteServerBusy); + remoteServersHook.bindClearStatusForServer(status.clearStatusForServer); + remoteServersHook.bindStopTunnelsForServer(tunnels.stopTunnelsForServer); + + const { activePage, openServer, openServersList, setSub } = useActivePage({ + remoteServers: remoteServersHook.remoteServers, + }); + + const scopeServerId = activePage.kind === "server" ? activePage.serverId : undefined; + const visibleLogRows = + scopeServerId && scopeToActiveServer + ? renderedLogRows.filter((row) => !row.serverId || row.serverId === scopeServerId) + : renderedLogRows; + + const activeServer = + activePage.kind === "server" + ? remoteServersHook.remoteServers.find((server) => server.id === activePage.serverId) + : undefined; + + return ( + + +
remoteServersHook.setRemoteAttachOpen(true)} + updateStatus={updates.updateStatus} + update={updates.availableUpdate} + updateProgress={updates.updateProgress} + onCheckUpdate={updates.checkForAppUpdate} + onOpenUpdate={() => updates.setUpdateDialogOpen(true)} + /> + + + appendLogRow(log.error("ui", message))}> + {activePage.kind === "servers" || !activeServer ? ( + remoteServersHook.setRemoteAttachOpen(true)} + /> + ) : ( + status.refreshRemoteServerStatus(activeServer)} + onRemove={() => remoteServersHook.setRemoteServerToRemove(activeServer)} + onStartBattlegroup={() => status.runRemoteBattlegroupAction(activeServer, "start")} + onStopBattlegroup={() => status.runRemoteBattlegroupAction(activeServer, "stop")} + onRestartBattlegroup={() => + status.runRemoteBattlegroupAction(activeServer, "restart") + } + onUpdateBattlegroup={() => status.runRemoteBattlegroupAction(activeServer, "update")} + onStartTunnel={tunnels.startServerTunnel} + onStartCustomTunnel={tunnels.startCustomTunnel} + onStopTunnel={tunnels.stopServerTunnel} + onOpenTunnel={tunnels.openServerTunnel} + onRefreshComponentLog={(component) => + componentActions.refreshRemoteComponentLog(activeServer, component) + } + onRestartComponent={(component) => + componentActions.restartRemoteComponent(activeServer, component) + } + appendLogRow={appendLogRow} + /> + )} + + + setLogPanelCollapsed((collapsed) => !collapsed)} + onToggleScope={setScopeToActiveServer} + /> + + { + remoteServersHook.setRemoteAttachOpen(open); + if (!open) remoteServersHook.setRemoteAttachError(null); + }} + onChange={remoteServersHook.setRemoteAttachForm} + onAttach={remoteServersHook.addRemoteServer} + /> + { + if (!open) remoteServersHook.setRemoteServerToRemove(null); + }} + onRemove={(server) => { + remoteServersHook.removeRemoteServer(server); + remoteServersHook.setRemoteServerToRemove(null); + }} + /> + + + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/AppErrorBoundary.tsx b/docs/reference-repos/adainrivers/app/src/components/AppErrorBoundary.tsx new file mode 100644 index 0000000..95d7e5b --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/AppErrorBoundary.tsx @@ -0,0 +1,43 @@ +import { Component, type ErrorInfo, type ReactNode } from "react"; +import { Card, Flex, Heading, Text } from "@radix-ui/themes"; + +export type AppErrorBoundaryProps = { + onError: (message: string) => void; + children: ReactNode; +}; + +type AppErrorBoundaryState = { + error: string | null; +}; + +export default class AppErrorBoundary extends Component { + state: AppErrorBoundaryState = { error: null }; + + static getDerivedStateFromError(error: Error): AppErrorBoundaryState { + return { error: error.message }; + } + + componentDidCatch(error: Error, info: ErrorInfo) { + this.props.onError(`${error.message}\n${info.componentStack}`); + } + + render() { + if (this.state.error) { + return ( + + + UI Error + + The view failed to render. Details were written to the log window. + + + {this.state.error} + + + + ); + } + + return this.props.children; + } +} diff --git a/docs/reference-repos/adainrivers/app/src/components/dialogs/AboutDialog.tsx b/docs/reference-repos/adainrivers/app/src/components/dialogs/AboutDialog.tsx new file mode 100644 index 0000000..b43df81 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/dialogs/AboutDialog.tsx @@ -0,0 +1,94 @@ +import { useEffect, useState } from "react"; +import { getVersion } from "@tauri-apps/api/app"; +import { InfoCircledIcon } from "@radix-ui/react-icons"; +import { Button, Dialog, Flex, IconButton, Link, Text } from "@radix-ui/themes"; + +import { openExternal } from "../../services/tauri"; + +const REPO_URL = "https://github.com/adainrivers/dune-dedicated-server-manager"; +const ISSUES_URL = `${REPO_URL}/issues`; + +/** + * Small info button (sits next to "Check for updates") that opens an About + * modal showing the app version and links back to the project. Self-contained: + * owns its open state so it can be dropped into the header without prop + * threading. + */ +export default function AboutDialog() { + const [open, setOpen] = useState(false); + const [version, setVersion] = useState(null); + + // The bundled dune-server-service ships with the same version as the app, + // so this number identifies both. Fetched from the Tauri runtime rather than + // package.json so it reflects the actually-installed build. + useEffect(() => { + let active = true; + void getVersion() + .then((v) => { + if (active) setVersion(v); + }) + .catch(() => { + if (active) setVersion(null); + }); + return () => { + active = false; + }; + }, []); + + const openLink = (url: string) => () => { + void openExternal(url); + }; + + return ( + + + + + + + + About + + Dune Dedicated Server Manager + + + + + + Version + + + {version ?? "—"} + + + + + + Repository + + { e.preventDefault(); openLink(REPO_URL)(); }}> + GitHub + + + + + + Found a bug? + + { e.preventDefault(); openLink(ISSUES_URL)(); }}> + Report an issue + + + + + + + + + + + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/dialogs/RemoteAttachDialog.tsx b/docs/reference-repos/adainrivers/app/src/components/dialogs/RemoteAttachDialog.tsx new file mode 100644 index 0000000..6e120be --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/dialogs/RemoteAttachDialog.tsx @@ -0,0 +1,134 @@ +import { Button, Callout, Dialog, Flex, Grid, TextField } from "@radix-ui/themes"; +import { ExclamationTriangleIcon } from "@radix-ui/react-icons"; + +import { openFileDialog, type PreflightCheck } from "../../services/tauri"; +import type { RemoteAttachForm } from "../../types/ui"; +import ActionButton from "../ui/ActionButton"; +import Field from "../ui/Field"; + +export type RemoteAttachDialogProps = { + open: boolean; + form: RemoteAttachForm; + running: boolean; + errorMessage?: string | null; + preflight?: PreflightCheck | null; + onOpenChange: (open: boolean) => void; + onChange: (form: RemoteAttachForm) => void; + onAttach: () => void; +}; + +export default function RemoteAttachDialog({ + open, + form, + running, + errorMessage, + preflight, + onOpenChange, + onChange, + onAttach, +}: RemoteAttachDialogProps) { + const canAttach = + form.host.trim().length > 0 && + form.user.trim().length > 0 && + form.keyPath.trim().length > 0 && + form.port > 0 && + !running; + return ( + + + Add Remote Server + + Connect over SSH and detect existing Dune battlegroups. Vendor wrapper commands + always execute as dune; if you log in as root we drop into dune via + sudo automatically. + + + + onChange({ ...form, host: event.target.value })} + /> + + + + onChange({ ...form, user: event.target.value })} + /> + + + { + const parsed = parseInt(event.target.value, 10); + onChange({ ...form, port: isNaN(parsed) ? 22 : parsed }); + }} + /> + + + + + onChange({ ...form, keyPath: event.target.value })} + /> + + + + {errorMessage ? ( + + + + + {errorMessage} + + ) : null} + {preflight && !errorMessage ? ( + + + Preflight passed: SSH ok, sudo to dune ok, dune passwordless sudo ok. + + + ) : null} + + + + + + + Detect and Add + + + + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/dialogs/RemoveRemoteServerDialog.tsx b/docs/reference-repos/adainrivers/app/src/components/dialogs/RemoveRemoteServerDialog.tsx new file mode 100644 index 0000000..2178cc7 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/dialogs/RemoveRemoteServerDialog.tsx @@ -0,0 +1,45 @@ +import { AlertDialog, Box, Button, Flex } from "@radix-ui/themes"; + +import type { RemoteServerRecord } from "../../types/server"; +import Metric from "../ui/Metric"; + +export type RemoveRemoteServerDialogProps = { + server: RemoteServerRecord | null; + onOpenChange: (open: boolean) => void; + onRemove: (server: RemoteServerRecord) => void; +}; + +export default function RemoveRemoteServerDialog({ + server, + onOpenChange, + onRemove, +}: RemoveRemoteServerDialogProps) { + return ( + + + Forget Remote Server + + This only removes the saved server entry from this app. The remote host and Dune battlegroup will not be changed. + + {server ? ( + + + + + ) : null} + + + + + + + + + + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/dialogs/UpdateDialog.tsx b/docs/reference-repos/adainrivers/app/src/components/dialogs/UpdateDialog.tsx new file mode 100644 index 0000000..dc744b9 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/dialogs/UpdateDialog.tsx @@ -0,0 +1,101 @@ +import type { ComponentPropsWithoutRef } from "react"; +import Markdown from "markdown-to-jsx"; +import { AlertDialog, Box, Button, Flex, Link, Text } from "@radix-ui/themes"; + +import { openExternal } from "../../services/tauri"; +import type { Update } from "../../services/updater"; +import type { UpdateStatus } from "../../types/update"; + +const RELEASES_URL = "https://github.com/adainrivers/dune-dedicated-server-manager/releases"; + +// Links inside the release notes must open in the system browser, not navigate +// the Tauri webview away from the app. +function NotesLink({ href, children }: ComponentPropsWithoutRef<"a">) { + return ( + { + e.preventDefault(); + if (href) void openExternal(href); + }} + > + {children} + + ); +} + +export type UpdateDialogProps = { + open: boolean; + update: Update | null; + status: UpdateStatus; + progress: string | null; + onOpenChange: (open: boolean) => void; + onInstall: () => void; +}; + +export default function UpdateDialog({ + open, + update, + status, + progress, + onOpenChange, + onInstall, +}: UpdateDialogProps) { + const busy = status === "installing" || status === "relaunching"; + return ( + + + Install app update? + + {update + ? `Version ${update.version} is available. The app will download the signed installer, install it, and relaunch.` + : "No update is currently selected."} + + {update?.body ? ( + + + What's new + + {/* Render the release notes as markdown, bounded with scroll so a + long changelog can never push the dialog past the viewport. */} + + + {update.body} + + + + { + e.preventDefault(); + void openExternal(RELEASES_URL); + }} + > + Full release notes + + + + ) : null} + {progress ? ( + + {progress} + + ) : null} + + + + + + + + + + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/layout/Header.tsx b/docs/reference-repos/adainrivers/app/src/components/layout/Header.tsx new file mode 100644 index 0000000..d8a4eab --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/layout/Header.tsx @@ -0,0 +1,75 @@ +import { Flex } from "@radix-ui/themes"; + +import type { Update } from "../../services/updater"; +import type { RemoteServerRecord, RemoteServerStatus } from "../../types/server"; +import type { ActivePage } from "../../types/ui"; +import type { UpdateStatus } from "../../types/update"; +import TopNav from "./TopNav"; +import UpdateHeaderControl from "./UpdateHeaderControl"; + +export type HeaderProps = { + activePage: ActivePage; + servers: RemoteServerRecord[]; + statuses: Record; + statusErrors: Record; + busyMap: Record; + onOpenServersList: () => void; + onOpenServer: (serverId: string) => void; + onAddServer: () => void; + updateStatus: UpdateStatus; + update: Update | null; + updateProgress: string | null; + onCheckUpdate: () => void; + onOpenUpdate: () => void; +}; + +export default function Header({ + activePage, + servers, + statuses, + statusErrors, + busyMap, + onOpenServersList, + onOpenServer, + onAddServer, + updateStatus, + update, + updateProgress, + onCheckUpdate, + onOpenUpdate, +}: HeaderProps) { + return ( + +
+ + + + D + + + Dune Dedicated Server Manager + Operator console + + + + + +
+
+ ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/layout/TopNav.tsx b/docs/reference-repos/adainrivers/app/src/components/layout/TopNav.tsx new file mode 100644 index 0000000..298e21a --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/layout/TopNav.tsx @@ -0,0 +1,83 @@ +import { TabNav, Tooltip } from "@radix-ui/themes"; +import { PlusIcon } from "@radix-ui/react-icons"; + +import type { RemoteServerRecord, RemoteServerStatus } from "../../types/server"; +import type { ActivePage } from "../../types/ui"; +import { resolveServerStatus } from "../../utils/remote-server"; + +export type TopNavProps = { + activePage: ActivePage; + servers: RemoteServerRecord[]; + statuses: Record; + statusErrors: Record; + busyMap: Record; + onOpenServersList: () => void; + onOpenServer: (serverId: string) => void; + onAddServer: () => void; +}; + +export default function TopNav({ + activePage, + servers, + statuses, + statusErrors, + busyMap, + onOpenServersList, + onOpenServer, + onAddServer, +}: TopNavProps) { + const serversActive = activePage.kind === "servers"; + const activeServerId = activePage.kind === "server" ? activePage.serverId : null; + return ( + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/layout/UpdateHeaderControl.tsx b/docs/reference-repos/adainrivers/app/src/components/layout/UpdateHeaderControl.tsx new file mode 100644 index 0000000..b8cd2e3 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/layout/UpdateHeaderControl.tsx @@ -0,0 +1,36 @@ +import { Badge, Button, Flex } from "@radix-ui/themes"; + +import type { Update } from "../../services/updater"; +import type { UpdateStatus } from "../../types/update"; +import { updateLabel, updateTone } from "../../utils/formatting"; +import AboutDialog from "../dialogs/AboutDialog"; + +export type UpdateHeaderControlProps = { + status: UpdateStatus; + update: Update | null; + progress: string | null; + onCheck: () => void; + onOpenUpdate: () => void; +}; + +export default function UpdateHeaderControl({ + status, + update, + progress, + onCheck, + onOpenUpdate, +}: UpdateHeaderControlProps) { + const busy = status === "checking" || status === "installing" || status === "relaunching"; + const hasUpdate = Boolean(update); + return ( + + + {updateLabel(status, update, progress)} + + + + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/logs/LogWindow.tsx b/docs/reference-repos/adainrivers/app/src/components/logs/LogWindow.tsx new file mode 100644 index 0000000..d75c67b --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/logs/LogWindow.tsx @@ -0,0 +1,185 @@ +import { useEffect, useLayoutEffect, useRef, useState } from "react"; +import { Box, Flex, Grid, Select, Text, Tooltip } from "@radix-ui/themes"; +import { + ChevronLeftIcon, + ChevronRightIcon, + FilePlusIcon, + TrashIcon, +} from "@radix-ui/react-icons"; + +import { getLogsFolder, openLogsFolder } from "../../services/tauri"; +import type { LogLevelFilter, LogRow } from "../../types/log"; + +export type LogWindowProps = { + rows: LogRow[]; + level: LogLevelFilter; + collapsed: boolean; + scopedToServer: boolean; + canScopeToServer: boolean; + onLevelChange: (level: LogLevelFilter) => void; + onClear: () => void; + onToggleCollapsed: () => void; + onToggleScope: (next: boolean) => void; +}; + +export default function LogWindow({ + rows, + level, + collapsed, + scopedToServer, + canScopeToServer, + onLevelChange, + onClear, + onToggleCollapsed, + onToggleScope, +}: LogWindowProps) { + const bodyRef = useRef(null); + const stickToBottomRef = useRef(true); + const [logsFolder, setLogsFolder] = useState(""); + + useLayoutEffect(() => { + const body = bodyRef.current; + if (!body) return; + if (stickToBottomRef.current) { + body.scrollTop = body.scrollHeight; + } + }, [rows]); + + useEffect(() => { + void getLogsFolder() + .then(setLogsFolder) + .catch(() => undefined); + }, []); + + const latestLevel = rows.length > 0 ? rows[rows.length - 1].level : "info"; + + if (collapsed) { + return ( + + ); + } + + return ( + + ); +} diff --git a/docs/reference-repos/adainrivers/app/src/components/management/AdminTab.tsx b/docs/reference-repos/adainrivers/app/src/components/management/AdminTab.tsx new file mode 100644 index 0000000..4455cb5 --- /dev/null +++ b/docs/reference-repos/adainrivers/app/src/components/management/AdminTab.tsx @@ -0,0 +1,789 @@ +import { useCallback, useEffect, useMemo, useRef, useState } from "react"; +import { + AlertDialog, + Badge, + Box, + Button, + Checkbox, + Flex, + Select, + Table, + Text, + TextArea, + TextField, +} from "@radix-ui/themes"; + +import { managementApi } from "../../services/management"; +import type { + Category, + CommandSpec, + FieldSpec, + HistoryDto, + PublishResultDto, +} from "../../types/management"; +import { formatTime } from "../../utils/formatting"; +import Combobox from "./Combobox"; + +export type AdminTabPrefill = { + commandId: string; + values: Record; +} | null; + +export type AdminTabProps = { + tunnelId: string; + prefill?: AdminTabPrefill; + onPrefillConsumed?: () => void; +}; + +const CATEGORY_LABEL: Record = { + items: "Inventory", + player: "Player ops", + progression: "Progression", + movement: "Teleport & spawn", + broadcast: "Broadcast", + journey: "Story journey", + exec: "Server scripts", +}; + +const CATEGORY_ORDER: Category[] = [ + "broadcast", + "items", + "player", + "progression", + "movement", + "journey", + "exec", +]; + +const CLIENT_DEFAULTS: Record = { + Quantity: 1, + Durability: 1.0, + WaterAmount: 1_000_000, + Experience: 1000, + Level: 1, + SkillPoints: 0, + BroadcastType: "Generic", + BroadcastDuration: 30, + Persistent: 1.0, +}; + +function applyDefaults(spec: CommandSpec): Record { + const out: Record = {}; + for (const field of spec.fields) { + if (CLIENT_DEFAULTS[field.key] !== undefined) { + out[field.key] = CLIENT_DEFAULTS[field.key]; + } else if (field.default !== undefined && field.default !== null) { + out[field.key] = field.default; + } + } + return out; +} + +export default function AdminTab({ tunnelId, prefill, onPrefillConsumed }: AdminTabProps) { + const [commands, setCommands] = useState([]); + const [selected, setSelected] = useState(null); + const [values, setValues] = useState>({}); + const [history, setHistory] = useState([]); + const [busy, setBusy] = useState(false); + const [result, setResult] = useState(null); + const [error, setError] = useState(null); + const [confirmOpen, setConfirmOpen] = useState(false); + const appliedRef = useRef<{ selectedId: string; prefillFp: string | null } | null>(null); + // Templates available for the currently-picked vehicle (SpawnVehicleAt). + // Populated whenever values.ClassName changes so TemplateName renders as a + // proper combobox of valid options instead of a free-text field. + const [vehicleTemplates, setVehicleTemplates] = useState([]); + + const refreshHistory = useCallback(async () => { + try { + const list = await managementApi.history(tunnelId, 30); + setHistory(list); + } catch (err) { + setError(String(err)); + } + }, [tunnelId]); + + useEffect(() => { + managementApi + .listCommands(tunnelId) + .then(setCommands) + .catch((err) => setError(String(err))); + void refreshHistory(); + }, [tunnelId, refreshHistory]); + + useEffect(() => { + // Reset values + apply prefill exactly once per (selected, prefill) pair. + // The earlier two-effect version raced; the single-effect version still + // clobbered prefill on the next render after onPrefillConsumed cleared it + // because the [prefill] dep change re-ran the defaults reset. Track what + // we've already applied so post-consumption re-renders are a no-op. + if (!selected) { + appliedRef.current = null; + return; + } + const prefillFp = + prefill && prefill.commandId === selected.id ? JSON.stringify(prefill) : null; + const current = appliedRef.current; + + if (!current || current.selectedId !== selected.id) { + // Brand new command pick (sidebar click or first prefill into a new command). + if (prefillFp) { + setValues({ ...applyDefaults(selected), ...(prefill?.values ?? {}) }); + onPrefillConsumed?.(); + } else { + setValues(applyDefaults(selected)); + } + setResult(null); + appliedRef.current = { selectedId: selected.id, prefillFp }; + return; + } + + // Same command. Only act if a NEW prefill arrived for it. + if (prefillFp && prefillFp !== current.prefillFp) { + setValues((prev) => ({ ...prev, ...(prefill?.values ?? {}) })); + setResult(null); + onPrefillConsumed?.(); + appliedRef.current = { selectedId: selected.id, prefillFp }; + } + // Otherwise the prefill was cleared after we consumed it — leave values alone. + }, [selected, prefill, onPrefillConsumed]); + + useEffect(() => { + // If a prefill arrives for a command different from what's currently + // selected, switch the sidebar to that command. The effect above will + // then notice prefill.commandId === selected.id and apply the values. + if (!prefill || commands.length === 0) return; + if (selected?.id === prefill.commandId) return; + const target = commands.find((c) => c.id === prefill.commandId); + if (!target) return; + setSelected(target); + }, [prefill, commands, selected?.id]); + + useEffect(() => { + // For SpawnVehicleAt, look up the templates of the picked vehicle so the + // TemplateName field can render its real options. + const cls = + selected?.id === "SpawnVehicleAt" && typeof values.ClassName === "string" + ? (values.ClassName as string).trim() + : ""; + if (!cls) { + setVehicleTemplates([]); + return; + } + let cancelled = false; + (async () => { + try { + const matches = await managementApi.searchVehicles(tunnelId, cls, 10); + const hit = matches.find((v) => v.id === cls || v.actor_class === cls); + const templates = hit?.templates ?? []; + if (cancelled) return; + setVehicleTemplates(templates); + // If the current TemplateName isn't valid for this vehicle, auto-pick + // the first available one. Keeps the form submittable without the user + // having to know that TreadWheel doesn't carry a T0. + if (templates.length > 0) { + setValues((prev) => { + const current = typeof prev.TemplateName === "string" ? prev.TemplateName : ""; + if (current && templates.includes(current)) return prev; + return { ...prev, TemplateName: templates[0] }; + }); + } + } catch { + if (!cancelled) setVehicleTemplates([]); + } + })(); + return () => { + cancelled = true; + }; + }, [selected?.id, values.ClassName, tunnelId]); + + const grouped = useMemo(() => groupByCategory(commands), [commands]); + + const doPublish = useCallback(async () => { + if (!selected) return; + setBusy(true); + setError(null); + setResult(null); + try { + const out = await managementApi.publish(tunnelId, selected.id, values); + setResult(out); + await refreshHistory(); + } catch (err) { + setError(String(err)); + } finally { + setBusy(false); + } + }, [selected, tunnelId, values, refreshHistory]); + + const publish = useCallback(() => { + if (!selected) return; + if (selected.destructive) { + setConfirmOpen(true); + } else { + void doPublish(); + } + }, [selected, doPublish]); + + return ( + + + + Commands + + {CATEGORY_ORDER.map((cat) => { + const specs = grouped[cat]; + if (!specs || specs.length === 0) return null; + return ( + + + {CATEGORY_LABEL[cat] ?? cat} + + + {specs.map((spec) => ( + + ))} + + + ); + })} + + + {selected ? ( + + + + {selected.label} + + {selected.destructive ? destructive : null} + + + {selected.describe} + + + {visibleFields(selected, values).map((field) => ( + setValues((prev) => ({ ...prev, [field.key]: v }))} + tunnelId={tunnelId} + vehicleTemplates={vehicleTemplates} + /> + ))} + + {selected.id === "SpawnVehicleAt" ? ( + + setValues((prev) => ({ ...prev, X: loc.x, Y: loc.y, Z: loc.z })) + } + /> + ) : null} + + + {result ? ( + {result.ok ? "ok" : "failed"} + ) : null} + + {result && !result.ok && result.error ? ( + + {result.error} + + ) : null} + {result?.output ? ( + + {result.output} + + ) : null} + {error ? ( + + {error} + + ) : null} + + ) : ( + Select a command on the left. + )} + + + + Recent publishes + + + + + Cmd + OK + When + + + + {history.map((h) => ( + + + {h.command} + + + {h.ok ? "ok" : "fail"} + + + {formatTime(h.createdAt)} + + + ))} + + + + + + + Run {selected?.label}? + + This command is destructive and cannot be undone. {selected?.describe} + + + + + + + + + + + ); +} + +function groupByCategory(specs: CommandSpec[]): Record { + const out: Record = {}; + for (const spec of specs) { + if (!out[spec.category]) out[spec.category] = []; + out[spec.category].push(spec); + } + return out; +} + +function compareText(a: string | undefined | null, b: string | undefined | null): number { + return (a || "").localeCompare(b || "", undefined, { sensitivity: "base", numeric: true }); +} + +function comparePlayers(a: any, b: any): number { + const aOnline = String(a.online || "").toLowerCase() === "online"; + const bOnline = String(b.online || "").toLowerCase() === "online"; + if (aOnline !== bOnline) return aOnline ? -1 : 1; + return compareText(a.name || a.flsId, b.name || b.flsId); +} + +function sortCommandOptions(kind: ComboboxKind, options: any[]): any[] { + const rows = [...options]; + if (kind === "players") return rows.sort(comparePlayers); + if (kind === "vehicles") return rows.sort((a, b) => compareText(a.id, b.id)); + return rows.sort((a, b) => compareText(a.name || a.id, b.name || b.id)); +} + +function UsePlayerPositionButton({ + tunnelId, + playerId, + onLocation, +}: { + tunnelId: string; + playerId: string | undefined; + onLocation: (loc: { x: number; y: number; z: number }) => void; +}) { + const [busy, setBusy] = useState(false); + const [error, setError] = useState(null); + const enabled = !!playerId && !busy; + + const click = useCallback(async () => { + if (!playerId) return; + setBusy(true); + setError(null); + try { + const loc = await managementApi.playerLocation(tunnelId, playerId); + onLocation(loc); + } catch (err) { + // Backend wraps proxy errors as `GET /path -> STATUS: {"error":"…"}`. + // Pull out the inner `error` field for a readable message; fall back to raw. + const raw = String(err); + let nice = raw; + const bodyStart = raw.indexOf("{"); + if (bodyStart >= 0) { + try { + const obj = JSON.parse(raw.slice(bodyStart)); + if (obj && typeof obj.error === "string") nice = obj.error; + } catch { + // leave nice as raw + } + } + setError(nice); + } finally { + setBusy(false); + } + }, [tunnelId, playerId, onLocation]); + + return ( + + + {!playerId ? ( + + (pick a player first) + + ) : null} + {error ? ( + + {error} + + ) : null} + + ); +} + +type ComboboxKind = "items" | "vehicles" | "players" | "skill-modules"; + +function comboboxKindFor(fieldKey: string): ComboboxKind | null { + switch (fieldKey) { + case "ItemName": + return "items"; + case "ClassName": + return "vehicles"; + case "PlayerId": + return "players"; + case "Module": + return "skill-modules"; + default: + return null; + } +} + +function FieldInput({ + field, + value, + onChange, + tunnelId, + vehicleTemplates, +}: { + field: FieldSpec; + value: unknown; + onChange: (v: unknown) => void; + tunnelId: string; + vehicleTemplates: string[]; +}) { + const comboKind = comboboxKindFor(field.key); + const templateMode = field.key === "TemplateName" && vehicleTemplates.length > 0; + return ( + + + + {field.label} + {field.required ? " *" : ""} + + {field.helper ? ( + + {field.helper} + + ) : null} + + + {templateMode ? ( + + ) : comboKind ? ( + + ) : ( + renderInput(field, value, onChange) + )} + + + ); +} + +function TemplateCombobox({ + value, + onPick, + templates, +}: { + value: string; + onPick: (v: unknown) => void; + templates: string[]; +}) { + const loadOptions = useCallback( + async (query: string) => { + const q = query.trim().toLowerCase(); + const filtered = q + ? templates.filter((t) => t.toLowerCase().includes(q)) + : templates; + return [...filtered].sort(compareText).map((name) => ({ name })); + }, + [templates], + ); + return ( + o.name} + resolveLabel={async (id) => id} + renderOption={(o: { name: string }) => ( + {o.name} + )} + placeholder="Pick a template…" + searchPlaceholder="Filter templates…" + /> + ); +} + +/// Filters the spec's field list down to what's relevant for the current +/// values. Today only ServiceBroadcast has conditional fields — Generic +/// hides the shutdown-specific knobs, ServerShutdown hides Generic-only +/// fields, and a `ShouldCancel=true` hides everything except the cancel +/// toggle itself. +function visibleFields( + spec: CommandSpec, + values: Record, +): FieldSpec[] { + if (spec.id !== "ServiceBroadcast") return [...spec.fields]; + const broadcastType = (values.BroadcastType as string) || "Generic"; + const shouldCancel = values.ShouldCancel === true; + const GENERIC_ONLY = new Set(["Title", "Body"]); + const SHUTDOWN_ONLY = new Set([ + "ShutdownType", + "ShutdownDuration", + "BroadcastFrequency", + "ShouldCancel", + ]); + return spec.fields.filter((field) => { + if (field.key === "BroadcastType") return true; + if (broadcastType === "Generic") { + if (SHUTDOWN_ONLY.has(field.key)) return false; + return true; + } + // ServerShutdown branch + if (GENERIC_ONLY.has(field.key)) return false; + if (shouldCancel && field.key !== "ShouldCancel") return false; + return true; + }); +} + +function renderInput(field: FieldSpec, value: unknown, onChange: (v: unknown) => void) { + const strValue = value === undefined || value === null ? "" : String(value); + if (field.kind === "select" && field.options) { + return ( + + + + {field.options.map((opt) => ( + + {opt.label} + + ))} + + + ); + } + if (field.kind === "text") { + return