diff --git a/.changeset/README.md b/.changeset/README.md new file mode 100644 index 0000000..1f2de68 --- /dev/null +++ b/.changeset/README.md @@ -0,0 +1 @@ +# Changesets diff --git a/.changeset/config.json b/.changeset/config.json new file mode 100644 index 0000000..edef2eb --- /dev/null +++ b/.changeset/config.json @@ -0,0 +1,11 @@ +{ + "$schema": "https://unpkg.com/@changesets/config@3.0.3/schema.json", + "changelog": "@changesets/cli/changelog", + "commit": false, + "fixed": [], + "linked": [], + "access": "restricted", + "baseBranch": "main", + "updateInternalDependencies": "patch", + "ignore": [] +} diff --git a/.github/FUNDING.yml b/.github/FUNDING.yml new file mode 100644 index 0000000..e7e3749 --- /dev/null +++ b/.github/FUNDING.yml @@ -0,0 +1,2 @@ +github: [HuakunShen] +buy_me_a_coffee: huakun diff --git a/.github/ISSUE_TEMPLATE/bug_report.yml b/.github/ISSUE_TEMPLATE/bug_report.yml new file mode 100644 index 0000000..68646db --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug_report.yml @@ -0,0 +1,43 @@ +name: Bug Report +description: File a bug report. +title: "[Bug]: " +labels: ["bug", "triage"] +body: + - type: markdown + attributes: + value: | + Thanks for taking the time to fill out this bug report! + - type: textarea + id: description + attributes: + label: Describe the bug + description: A clear description of what the bug is. Include screenshots if applicable. If you intend to submit a PR for this issue, tell us in the description. Thanks! + placeholder: Bug description + validations: + required: true + + - type: textarea + id: expected-behavior + attributes: + label: Expected behavior + description: A clear description of what you expected to happen. + validations: + required: true + + - type: textarea + id: system-info + attributes: + label: System Info + description: Output of `npx envinfo --system --npmPackages @kksh/api --binaries --browsers` + render: bash + placeholder: System, Binaries, Browsers + validations: + required: true + + - type: checkboxes + id: contributes + attributes: + label: Contributes + options: + - label: I am willing to submit a PR to fix this issue + - label: I am willing to submit a PR with failing tests \ No newline at end of file diff --git a/.github/ISSUE_TEMPLATE/config.yml b/.github/ISSUE_TEMPLATE/config.yml new file mode 100644 index 0000000..44f8b92 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/config.yml @@ -0,0 +1,8 @@ +blank_issues_enabled: false +contact_links: + - name: GitHub Discussions + url: https://github.com/kunkunsh/kunkun/discussions + about: Please ask and answer questions here. + - name: 💬 Discord + url: https://discord.gg/7dzw3TYeTU + about: Please ask and answer questions here. diff --git a/.github/ISSUE_TEMPLATE/feature_request.yml b/.github/ISSUE_TEMPLATE/feature_request.yml new file mode 100644 index 0000000..bba670d --- /dev/null +++ b/.github/ISSUE_TEMPLATE/feature_request.yml @@ -0,0 +1,44 @@ +name: 💡 Feature Request +title: "[feat] " +description: Suggest an idea +labels: ["type: feature request"] + +body: + - type: textarea + id: problem + attributes: + label: Describe the problem + description: A clear description of the problem this feature would solve + placeholder: "I'm always frustrated when..." + validations: + required: true + + - type: textarea + id: solution + attributes: + label: "Describe the solution you'd like" + description: A clear description of what change you would like + placeholder: "I would like to..." + validations: + required: true + + - type: textarea + id: alternatives + attributes: + label: Alternatives considered + description: "Any alternative solutions you've considered" + + - type: textarea + id: context + attributes: + label: Additional context + description: Add any other context about the problem here. + + - type: checkboxes + id: contributes + attributes: + label: Contributes + options: + - label: I am willing to submit a PR to implement this feature + + diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..8a2fa15 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,39 @@ +name: CI +on: + push: + workflow_dispatch: + pull_request: + branches: + - main + +jobs: + build-test: + strategy: + matrix: + os: [ubuntu-24.04, macos-14, windows-latest] + runs-on: ${{ matrix.os }} + steps: + - uses: actions/checkout@v4 + with: + submodules: "true" + - uses: pnpm/action-setup@v4 + - name: Setup Node + uses: actions/setup-node@v4 + with: + node-version: 22 + cache: "pnpm" + cache-dependency-path: ./pnpm-lock.yaml + - uses: oven-sh/setup-bun@v1 + with: + bun-version: 1.1.27 + - uses: denoland/setup-deno@v2 + with: + deno-version: v2.x + - name: Install Dependencies + run: pnpm install + - name: Setup + run: pnpm prepare + - name: Build + run: pnpm build + - name: Test + run: pnpm test diff --git a/.github/workflows/jsr-publish.yml b/.github/workflows/jsr-publish.yml new file mode 100644 index 0000000..062b25a --- /dev/null +++ b/.github/workflows/jsr-publish.yml @@ -0,0 +1,22 @@ +name: JSR Publish +on: + workflow_dispatch: + +jobs: + publish: + runs-on: ubuntu-latest + + permissions: + contents: read + id-token: write + + steps: + - uses: actions/checkout@v4 + - uses: denoland/setup-deno@v2 + with: + deno-version: v2.x + - name: Publish package + working-directory: ./packages/api + run: | + deno install + npx jsr publish --allow-slow-types diff --git a/.github/workflows/manifest-schema-upload.yml b/.github/workflows/manifest-schema-upload.yml new file mode 100644 index 0000000..3aa0c8f --- /dev/null +++ b/.github/workflows/manifest-schema-upload.yml @@ -0,0 +1,33 @@ +name: Update Extension Manifest Schema + +on: + workflow_dispatch: + +jobs: + upload: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + with: + submodules: "true" + - uses: pnpm/action-setup@v4 + - name: Setup Node + uses: actions/setup-node@v4 + with: + node-version: 22 + cache: "pnpm" + cache-dependency-path: ./pnpm-lock.yaml + - uses: oven-sh/setup-bun@v1 + with: + bun-version: 1.1.27 + - uses: denoland/setup-deno@v2 + with: + deno-version: v2.x + - name: Install Dependencies + run: pnpm install + - name: Setup + run: pnpm prepare + - name: Update Schema + env: + SUPABASE_SERVICE_ROLE_KEY: ${{ secrets.SUPABASE_SERVICE_ROLE_KEY }} + run: pnpm --filter @kksh/schema upload-schema-to-supabase diff --git a/.gitignore b/.gitignore index 96fab4f..e2b56d1 100644 --- a/.gitignore +++ b/.gitignore @@ -36,3 +36,5 @@ yarn-error.log* # Misc .DS_Store *.pem +stats.html +target/ diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 0000000..a794889 --- /dev/null +++ b/.gitmodules @@ -0,0 +1,9 @@ +[submodule "vendors/applications-rs"] + path = vendors/applications-rs + url = https://github.com/HuakunShen/applications-rs.git +[submodule "vendors/tauri-plugin-network"] + path = vendors/tauri-plugin-network + url = https://github.com/HuakunShen/tauri-plugin-network.git +[submodule "vendors/tauri-plugin-system-info"] + path = vendors/tauri-plugin-system-info + url = https://github.com/HuakunShen/tauri-plugin-system-info.git diff --git a/.npmrc b/.npmrc new file mode 100644 index 0000000..41583e3 --- /dev/null +++ b/.npmrc @@ -0,0 +1 @@ +@jsr:registry=https://npm.jsr.io diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 0000000..77a58de --- /dev/null +++ b/.prettierignore @@ -0,0 +1,3 @@ +.svelte-kit/ +target/ +vendors \ No newline at end of file diff --git a/.vscode/extensions.json b/.vscode/extensions.json index 61343e9..63da6db 100644 --- a/.vscode/extensions.json +++ b/.vscode/extensions.json @@ -2,6 +2,7 @@ "recommendations": [ "svelte.svelte-vscode", "tauri-apps.tauri-vscode", - "rust-lang.rust-analyzer" + "rust-lang.rust-analyzer", + "denoland.vscode-deno" ] } diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 0000000..f7b5214 --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,45 @@ +# Contributing + +If you are interested in contributing to the project, please read the following guidelines. + +## Development + +### Prerequisites + +- [Node.js](https://nodejs.org/en) +- [pnpm](https://pnpm.io/) +- [Bun](https://bun.sh/) +- [Deno](https://deno.com/) +- [Rust](https://www.rust-lang.org/) +- [protobuf](https://grpc.io/docs/protoc-installation/) + - MacOS: `brew install protobuf` + - Linux: `sudo apt install -y protobuf-compiler` + - Windows: + ```powershell + choco install protoc + choco install openssl + ``` + Then configure the environment variables (yours may differ): + - `OPENSSL_DIR`: `C:\Program Files\OpenSSL-Win64` + - `OPENSSL_INCLUDE_DIR`: `C:\Program Files\OpenSSL-Win64\include` + - `OPENSSL_LIB_DIR`: `C:\Program Files\OpenSSL-Win64\lib` +- [cmake](https://cmake.org/) + - MacOS: `brew install cmake` + - Linux: `sudo apt install -y cmake` + +### Setup + +```bash +git clone https://github.com/kunkunsh/kunkun.git --recursive +pnpm install +pnpm prepare +``` + +### Run Desktop App + +```bash +pnpm --filter @kksh/desktop tauri dev +# or run it within the desktop app directory +cd apps/desktop +pnpm tauri dev +``` diff --git a/Cargo.lock b/Cargo.lock new file mode 100644 index 0000000..e5c93db --- /dev/null +++ b/Cargo.lock @@ -0,0 +1,8622 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 3 + +[[package]] +name = "addr2line" +version = "0.24.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dfbe277e56a376000877090da837660b4427aad530e3028d44e0bffe4f89a1c1" +dependencies = [ + "gimli", +] + +[[package]] +name = "adler2" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627" + +[[package]] +name = "aes" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0" +dependencies = [ + "cfg-if", + "cipher", + "cpufeatures", +] + +[[package]] +name = "ahash" +version = "0.7.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "891477e0c6a8957309ee5c45a6368af3ae14bb510732d2684ffa19af310920f9" +dependencies = [ + "getrandom 0.2.15", + "once_cell", + "version_check", +] + +[[package]] +name = "ahash" +version = "0.8.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e89da841a80418a9b391ebaea17f5c112ffaaa96f621d2c285b5174da76b9011" +dependencies = [ + "cfg-if", + "once_cell", + "version_check", + "zerocopy", +] + +[[package]] +name = "aho-corasick" +version = "1.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916" +dependencies = [ + "memchr", +] + +[[package]] +name = "aligned-vec" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4aa90d7ce82d4be67b64039a3d588d38dbcc6736577de4a847025ce5b0c468d1" + +[[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-tzdata" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e999941b234f3131b00bc13c22d06e8c5ff726d1b6318ac7eb276997bbb4fef0" + +[[package]] +name = "android_log-sys" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ecc8056bf6ab9892dcd53216c83d1597487d7dacac16c8df6b877d127df9937" + +[[package]] +name = "android_logger" +version = "0.14.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05b07e8e73d720a1f2e4b6014766e6039fd2e96a4fa44e2a78d0e1fa2ff49826" +dependencies = [ + "android_log-sys", + "env_filter", + "log", +] + +[[package]] +name = "android_system_properties" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311" +dependencies = [ + "libc", +] + +[[package]] +name = "anstream" +version = "0.6.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23a1e53f0f5d86382dafe1cf314783b2044280f406e7e1506368220ad11b1338" +dependencies = [ + "anstyle", + "anstyle-parse", + "anstyle-query", + "anstyle-wincon", + "colorchoice", + "is_terminal_polyfill", + "utf8parse", +] + +[[package]] +name = "anstyle" +version = "1.0.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55cc3b69f167a1ef2e161439aa98aed94e6028e5f9a59be9a6ffb47aef1651f9" + +[[package]] +name = "anstyle-parse" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b2d16507662817a6a20a9ea92df6652ee4f94f914589377d69f3b21bc5798a9" +dependencies = [ + "utf8parse", +] + +[[package]] +name = "anstyle-query" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "79947af37f4177cfead1110013d678905c37501914fba0efea834c3fe9a8d60c" +dependencies = [ + "windows-sys 0.59.0", +] + +[[package]] +name = "anstyle-wincon" +version = "3.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2109dbce0e72be3ec00bed26e6a7479ca384ad226efdd66db8fa2e3a38c83125" +dependencies = [ + "anstyle", + "windows-sys 0.59.0", +] + +[[package]] +name = "anyhow" +version = "1.0.92" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "74f37166d7d48a0284b99dd824694c26119c700b53bf0d1540cdb147dbdaaf13" + +[[package]] +name = "applications" +version = "0.2.3" +dependencies = [ + "anyhow", + "cocoa 0.25.0", + "core-foundation 0.9.4", + "glob", + "image", + "ini", + "lnk", + "objc", + "parselnk", + "plist", + "regex", + "serde", + "serde_derive", + "serde_json", + "tauri-icns", + "thiserror", + "walkdir", + "winreg 0.52.0", +] + +[[package]] +name = "arbitrary" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7d5a26814d8dcb93b0e5a0ff3c6d80a8843bafb21b39e8e18a6f05471870e110" +dependencies = [ + "derive_arbitrary", +] + +[[package]] +name = "arc-swap" +version = "1.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69f7f8c3906b62b754cd5326047894316021dcfe5a194c8ea52bdd94934a3457" + +[[package]] +name = "arg_enum_proc_macro" +version = "0.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ae92a5119aa49cdbcf6b9f893fe4e1d98b04ccbf82ee0584ad948a44a734dea" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "arrayvec" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50" + +[[package]] +name = "ashpd" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4d43c03d9e36dd40cab48435be0b09646da362c278223ca535493877b2c1dee9" +dependencies = [ + "enumflags2", + "futures-channel", + "futures-util", + "rand 0.8.5", + "raw-window-handle", + "serde", + "serde_repr", + "tokio", + "url", + "wayland-backend", + "wayland-client", + "wayland-protocols", + "zbus", +] + +[[package]] +name = "async-broadcast" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20cd0e2e25ea8e5f7e9df04578dc6cf5c83577fd09b1a46aaf5c85e1c33f2a7e" +dependencies = [ + "event-listener", + "event-listener-strategy", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "async-channel" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "89b47800b0be77592da0afd425cc03468052844aff33b84e33cc696f64e77b6a" +dependencies = [ + "concurrent-queue", + "event-listener-strategy", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "async-executor" +version = "1.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "30ca9a001c1e8ba5149f91a74362376cc6bc5b919d92d988668657bd570bdcec" +dependencies = [ + "async-task", + "concurrent-queue", + "fastrand", + "futures-lite", + "slab", +] + +[[package]] +name = "async-fs" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebcd09b382f40fcd159c2d695175b2ae620ffa5f3bd6f664131efff4e8b9e04a" +dependencies = [ + "async-lock", + "blocking", + "futures-lite", +] + +[[package]] +name = "async-io" +version = "2.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "444b0228950ee6501b3568d3c93bf1176a1fdbc3b758dcd9475046d30f4dc7e8" +dependencies = [ + "async-lock", + "cfg-if", + "concurrent-queue", + "futures-io", + "futures-lite", + "parking", + "polling 3.7.3", + "rustix", + "slab", + "tracing", + "windows-sys 0.59.0", +] + +[[package]] +name = "async-lock" +version = "3.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff6e472cdea888a4bd64f342f09b3f50e1886d32afe8df3d663c01140b811b18" +dependencies = [ + "event-listener", + "event-listener-strategy", + "pin-project-lite", +] + +[[package]] +name = "async-process" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63255f1dc2381611000436537bbedfe83183faa303a5a0edaf191edef06526bb" +dependencies = [ + "async-channel", + "async-io", + "async-lock", + "async-signal", + "async-task", + "blocking", + "cfg-if", + "event-listener", + "futures-lite", + "rustix", + "tracing", +] + +[[package]] +name = "async-recursion" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b43422f69d8ff38f95f1b2bb76517c91589a924d1559a0e935d7c8ce0274c11" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "async-signal" +version = "0.2.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "637e00349800c0bdf8bfc21ebbc0b6524abea702b0da4168ac00d070d0c0b9f3" +dependencies = [ + "async-io", + "async-lock", + "atomic-waker", + "cfg-if", + "futures-core", + "futures-io", + "rustix", + "signal-hook-registry", + "slab", + "windows-sys 0.59.0", +] + +[[package]] +name = "async-stream" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b5a71a6f37880a80d1d7f19efd781e4b5de42c88f0722cc13bcb6cc2cfe8476" +dependencies = [ + "async-stream-impl", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "async-stream-impl" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7c24de15d275a1ecfd47a380fb4d5ec9bfe0933f309ed5e705b775596a3574d" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "async-task" +version = "4.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b75356056920673b02621b35afd0f7dda9306d03c79a30f5c56c44cf256e3de" + +[[package]] +name = "async-trait" +version = "0.1.83" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "721cae7de5c34fbb2acd27e21e6d2cf7b886dce0c27388d46c4e6c47ea4318dd" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "atk" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b4af014b17dd80e8af9fa689b2d4a211ddba6eb583c1622f35d0cb543f6b17e4" +dependencies = [ + "atk-sys", + "glib", + "libc", +] + +[[package]] +name = "atk-sys" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "251e0b7d90e33e0ba930891a505a9a35ece37b2dd37a14f3ffc306c13b980009" +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.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26" + +[[package]] +name = "av1-grain" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6678909d8c5d46a42abcf571271e15fdbc0a225e3646cf23762cd415046c78bf" +dependencies = [ + "anyhow", + "arrayvec", + "log", + "nom", + "num-rational", + "v_frame", +] + +[[package]] +name = "avif-serialize" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e335041290c43101ca215eed6f43ec437eb5a42125573f600fc3fa42b9bddd62" +dependencies = [ + "arrayvec", +] + +[[package]] +name = "axum" +version = "0.6.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b829e4e32b91e643de6eafe82b1d90675f5874230191a4ffbc1b336dec4d6bf" +dependencies = [ + "async-trait", + "axum-core", + "bitflags 1.3.2", + "bytes", + "futures-util", + "http 0.2.12", + "http-body 0.4.6", + "hyper 0.14.31", + "itoa 1.0.11", + "matchit", + "memchr", + "mime", + "percent-encoding", + "pin-project-lite", + "rustversion", + "serde", + "serde_json", + "serde_path_to_error", + "serde_urlencoded", + "sync_wrapper 0.1.2", + "tokio", + "tower", + "tower-layer", + "tower-service", +] + +[[package]] +name = "axum-core" +version = "0.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "759fa577a247914fd3f7f76d62972792636412fbfd634cd452f6a385a74d2d2c" +dependencies = [ + "async-trait", + "bytes", + "futures-util", + "http 0.2.12", + "http-body 0.4.6", + "mime", + "rustversion", + "tower-layer", + "tower-service", +] + +[[package]] +name = "axum-extra" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ab90e7b70bea63a153137162affb6a0bce26b584c24a4c7885509783e2cf30b" +dependencies = [ + "axum", + "axum-core", + "bytes", + "futures-util", + "http 0.2.12", + "http-body 0.4.6", + "mime", + "pin-project-lite", + "serde", + "tokio", + "tower", + "tower-layer", + "tower-service", +] + +[[package]] +name = "axum-server" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "447f28c85900215cc1bea282f32d4a2f22d55c5a300afdfbc661c8d6a632e063" +dependencies = [ + "arc-swap", + "bytes", + "futures-util", + "http 0.2.12", + "http-body 0.4.6", + "hyper 0.14.31", + "pin-project-lite", + "rustls 0.21.12", + "rustls-pemfile 1.0.4", + "tokio", + "tokio-rustls 0.24.1", + "tower-service", +] + +[[package]] +name = "backtrace" +version = "0.3.74" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8d82cb332cdfaed17ae235a638438ac4d4839913cc2af585c3c6746e8f8bee1a" +dependencies = [ + "addr2line", + "cfg-if", + "libc", + "miniz_oxide", + "object", + "rustc-demangle", + "windows-targets 0.52.6", +] + +[[package]] +name = "base64" +version = "0.21.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" + +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + +[[package]] +name = "bit_field" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc827186963e592360843fb5ba4b973e145841266c1357f7180c43526f2e5b61" + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + +[[package]] +name = "bitflags" +version = "2.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b048fb63fd8b5923fc5aa7b340d8e156aec7ec02f0c78fa8a6ddc2613f6f71de" +dependencies = [ + "serde", +] + +[[package]] +name = "bitstream-io" +version = "2.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b81e1519b0d82120d2fd469d5bfb2919a9361c48b02d82d04befc1cdd2002452" + +[[package]] +name = "bitvec" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bc2832c24239b0141d5674bb9174f9d68a8b5b3f2753311927c172ca46f7e9c" +dependencies = [ + "funty", + "radium", + "tap", + "wyz", +] + +[[package]] +name = "block" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0d8c1fef690941d3e7788d328517591fecc684c084084702d6ff1641e993699a" + +[[package]] +name = "block-buffer" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" +dependencies = [ + "generic-array", +] + +[[package]] +name = "block2" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2c132eebf10f5cad5289222520a4a058514204aed6d791f1cf4fe8088b82d15f" +dependencies = [ + "objc2", +] + +[[package]] +name = "blocking" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "703f41c54fc768e63e091340b424302bb1c29ef4aa0c7f10fe849dfb114d29ea" +dependencies = [ + "async-channel", + "async-task", + "futures-io", + "futures-lite", + "piper", +] + +[[package]] +name = "borsh" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26d4d6dafc1a3bb54687538972158f07b2c948bc57d5890df22c0739098b3028" +dependencies = [ + "borsh-derive", + "cfg_aliases 0.1.1", +] + +[[package]] +name = "borsh-derive" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf4918709cc4dd777ad2b6303ed03cb37f3ca0ccede8c1b0d28ac6db8f4710e0" +dependencies = [ + "once_cell", + "proc-macro-crate 2.0.2", + "proc-macro2", + "quote", + "syn 2.0.86", + "syn_derive", +] + +[[package]] +name = "brotli" +version = "7.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc97b8f16f944bba54f0433f07e30be199b6dc2bd25937444bbad560bcea29bd" +dependencies = [ + "alloc-no-stdlib", + "alloc-stdlib", + "brotli-decompressor", +] + +[[package]] +name = "brotli-decompressor" +version = "4.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a45bd2e4095a8b518033b128020dd4a55aab1c0a381ba4404a472630f4bc362" +dependencies = [ + "alloc-no-stdlib", + "alloc-stdlib", +] + +[[package]] +name = "bstr" +version = "1.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "40723b8fb387abc38f4f4a37c09073622e41dd12327033091ef8950659e6dc0c" +dependencies = [ + "memchr", + "serde", +] + +[[package]] +name = "built" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c360505aed52b7ec96a3636c3f039d99103c37d1d9b4f7a8c743d3ea9ffcd03b" + +[[package]] +name = "bumpalo" +version = "3.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c" + +[[package]] +name = "byte-unit" +version = "5.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50af9a3004c52b9ca954e0bc87662f5c0a71a133af7c9ed6024b1e12b83cc292" +dependencies = [ + "rust_decimal", + "serde", + "utf8-width", +] + +[[package]] +name = "bytecheck" +version = "0.6.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23cdc57ce23ac53c931e88a43d06d070a6fd142f2617be5855eb75efc9beb1c2" +dependencies = [ + "bytecheck_derive", + "ptr_meta", + "simdutf8", +] + +[[package]] +name = "bytecheck_derive" +version = "0.6.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3db406d29fbcd95542e92559bed4d8ad92636d1ca8b3b72ede10b4bcc010e659" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "bytemuck" +version = "1.19.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8334215b81e418a0a7bdb8ef0849474f40bb10c8b71f1c4ed315cff49f32494d" + +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" + +[[package]] +name = "byteorder-lite" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f1fe948ff07f4bd06c30984e69f5b4899c516a3ef74f34df92a2df2ab535495" + +[[package]] +name = "bytes" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ac0150caa2ae65ca5bd83f25c7de183dea78d4d366469f148435e2acfbad0da" +dependencies = [ + "serde", +] + +[[package]] +name = "bzip2" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bdb116a6ef3f6c3698828873ad02c3014b3c85cadb88496095628e3ef1e347f8" +dependencies = [ + "bzip2-sys", + "libc", +] + +[[package]] +name = "bzip2-sys" +version = "0.1.11+1.0.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "736a955f3fa7875102d57c82b8cac37ec45224a07fd32d58f9f7a186b6cd4cdc" +dependencies = [ + "cc", + "libc", + "pkg-config", +] + +[[package]] +name = "cairo-rs" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ca26ef0159422fb77631dc9d17b102f253b876fe1586b03b803e63a309b4ee2" +dependencies = [ + "bitflags 2.6.0", + "cairo-sys-rs", + "glib", + "libc", + "once_cell", + "thiserror", +] + +[[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.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b96ec4966b5813e2c0507c1f86115c8c5abaadc3980879c3424042a02fd1ad3" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo-platform" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24b1f0365a6c6bb4020cd05806fd0d33c44d38046b8bd7f0e40814b9763cabfc" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo_metadata" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2d886547e41f740c616ae73108f6eb70afe6d940c7bc697cb30f13daec073037" +dependencies = [ + "camino", + "cargo-platform", + "semver", + "serde", + "serde_json", + "thiserror", +] + +[[package]] +name = "cargo_toml" +version = "0.17.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a969e13a7589e9e3e4207e153bae624ade2b5622fb4684a4923b23ec3d57719" +dependencies = [ + "serde", + "toml 0.8.2", +] + +[[package]] +name = "cc" +version = "1.1.33" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3788d6ac30243803df38a3e9991cf37e41210232916d41a8222ae378f912624" +dependencies = [ + "jobserver", + "libc", + "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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" + +[[package]] +name = "cfg_aliases" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd16c4719339c4530435d38e511904438d07cce7950afa3718a84ac36c10e89e" + +[[package]] +name = "cfg_aliases" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724" + +[[package]] +name = "chrono" +version = "0.4.38" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a21f936df1771bf62b77f047b726c4625ff2e8aa607c01ec06e5a05bd8463401" +dependencies = [ + "android-tzdata", + "iana-time-zone", + "js-sys", + "num-traits", + "serde", + "wasm-bindgen", + "windows-targets 0.52.6", +] + +[[package]] +name = "cipher" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad" +dependencies = [ + "crypto-common", + "inout", +] + +[[package]] +name = "clap" +version = "4.5.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b97f376d85a664d5837dbae44bf546e6477a679ff6610010f17276f686d867e8" +dependencies = [ + "clap_builder", +] + +[[package]] +name = "clap_builder" +version = "4.5.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19bc80abd44e4bed93ca373a0704ccbd1b710dc5749406201bb018272808dc54" +dependencies = [ + "anstream", + "anstyle", + "clap_lex", + "strsim 0.11.1", +] + +[[package]] +name = "clap_lex" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1462739cb27611015575c0c11df5df7601141071f07518d56fcc1be504cbec97" + +[[package]] +name = "clipboard-rs" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd61885ce9f906c60f40d384b132fed04685f51d3da3576d27b8e4f274b6100d" +dependencies = [ + "clipboard-win", + "image", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "x11rb", +] + +[[package]] +name = "clipboard-win" +version = "5.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15efe7a882b08f34e38556b14f2fb3daa98769d06c7f0c1b076dfd0d983bc892" +dependencies = [ + "error-code", + "windows-win", +] + +[[package]] +name = "cmake" +version = "0.1.51" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fb1e43aa7fd152b1f968787f7dbcdeb306d1867ff373c69955211876c053f91a" +dependencies = [ + "cc", +] + +[[package]] +name = "cocoa" +version = "0.24.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f425db7937052c684daec3bd6375c8abe2d146dca4b8b143d6db777c39138f3a" +dependencies = [ + "bitflags 1.3.2", + "block", + "cocoa-foundation 0.1.2", + "core-foundation 0.9.4", + "core-graphics 0.22.3", + "foreign-types 0.3.2", + "libc", + "objc", +] + +[[package]] +name = "cocoa" +version = "0.25.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6140449f97a6e97f9511815c5632d84c8aacf8ac271ad77c559218161a1373c" +dependencies = [ + "bitflags 1.3.2", + "block", + "cocoa-foundation 0.1.2", + "core-foundation 0.9.4", + "core-graphics 0.23.2", + "foreign-types 0.5.0", + "libc", + "objc", +] + +[[package]] +name = "cocoa" +version = "0.26.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f79398230a6e2c08f5c9760610eb6924b52aa9e7950a619602baba59dcbbdbb2" +dependencies = [ + "bitflags 2.6.0", + "block", + "cocoa-foundation 0.2.0", + "core-foundation 0.10.0", + "core-graphics 0.24.0", + "foreign-types 0.5.0", + "libc", + "objc", +] + +[[package]] +name = "cocoa-foundation" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8c6234cbb2e4c785b456c0644748b1ac416dd045799740356f8363dfe00c93f7" +dependencies = [ + "bitflags 1.3.2", + "block", + "core-foundation 0.9.4", + "core-graphics-types 0.1.3", + "libc", + "objc", +] + +[[package]] +name = "cocoa-foundation" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e14045fb83be07b5acf1c0884b2180461635b433455fa35d1cd6f17f1450679d" +dependencies = [ + "bitflags 2.6.0", + "block", + "core-foundation 0.10.0", + "core-graphics-types 0.2.0", + "libc", + "objc", +] + +[[package]] +name = "color_quant" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d7b894f5411737b7867f4827955924d7c254fc9f4d91a6aad6b097804b1018b" + +[[package]] +name = "colorchoice" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b63caa9aa9397e2d9480a9b13673856c78d8ac123288526c37d7839f2a86990" + +[[package]] +name = "colored" +version = "1.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a5f741c91823341bebf717d4c71bda820630ce065443b58bd1b7451af008355" +dependencies = [ + "is-terminal", + "lazy_static", + "winapi", +] + +[[package]] +name = "combine" +version = "4.6.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd" +dependencies = [ + "bytes", + "memchr", +] + +[[package]] +name = "concurrent-queue" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ca0197aee26d1ae37445ee532fefce43251d24cc7c166799f4d46817f1d3973" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "configparser" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fe1d7dcda7d1da79e444bdfba1465f2f849a58b07774e1df473ee77030cb47a7" + +[[package]] +name = "const-random" +version = "0.1.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87e00182fe74b066627d63b85fd550ac2998d4b0bd86bfed477a0ae4c7c71359" +dependencies = [ + "const-random-macro", +] + +[[package]] +name = "const-random-macro" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f9d839f2a20b0aee515dc581a6172f2321f96cab76c1a38a4c584a194955390e" +dependencies = [ + "getrandom 0.2.15", + "once_cell", + "tiny-keccak", +] + +[[package]] +name = "constant_time_eq" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c74b8349d32d297c9134b8c88677813a227df8f779daa29bfc29c183fe3dca6" + +[[package]] +name = "convert_case" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" + +[[package]] +name = "cookie" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ddef33a339a91ea89fb53151bd0a4689cfce27055c291dfa69945475d22c747" +dependencies = [ + "percent-encoding", + "time", + "version_check", +] + +[[package]] +name = "cookie_store" +version = "0.21.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4934e6b7e8419148b6ef56950d277af8561060b56afd59e2aadf98b59fce6baa" +dependencies = [ + "cookie", + "idna 0.5.0", + "log", + "publicsuffix", + "serde", + "serde_derive", + "serde_json", + "time", + "url", +] + +[[package]] +name = "core-foundation" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "core-foundation" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b55271e5c8c478ad3f38ad24ef34923091e0548492a266d19b3c0b4d82574c63" +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.22.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2581bbab3b8ffc6fcbd550bf46c355135d16e9ff2a6ea032ad6b9bf1d7efe4fb" +dependencies = [ + "bitflags 1.3.2", + "core-foundation 0.9.4", + "core-graphics-types 0.1.3", + "foreign-types 0.3.2", + "libc", +] + +[[package]] +name = "core-graphics" +version = "0.23.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c07782be35f9e1140080c6b96f0d44b739e2278479f64e02fdab4e32dfd8b081" +dependencies = [ + "bitflags 1.3.2", + "core-foundation 0.9.4", + "core-graphics-types 0.1.3", + "foreign-types 0.5.0", + "libc", +] + +[[package]] +name = "core-graphics" +version = "0.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa95a34622365fa5bbf40b20b75dba8dfa8c94c734aea8ac9a5ca38af14316f1" +dependencies = [ + "bitflags 2.6.0", + "core-foundation 0.10.0", + "core-graphics-types 0.2.0", + "foreign-types 0.5.0", + "libc", +] + +[[package]] +name = "core-graphics-types" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "45390e6114f68f718cc7a830514a96f903cccd70d02a8f6d9f643ac4ba45afaf" +dependencies = [ + "bitflags 1.3.2", + "core-foundation 0.9.4", + "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.6.0", + "core-foundation 0.10.0", + "libc", +] + +[[package]] +name = "cpufeatures" +version = "0.2.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "608697df725056feaccfa42cffdaeeec3fccc4ffc38358ecd19b243e716a78e0" +dependencies = [ + "libc", +] + +[[package]] +name = "crc" +version = "3.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69e6e4d7b33a94f0991c26729976b10ebde1d34c3ee82408fb536164fa10d636" +dependencies = [ + "crc-catalog", +] + +[[package]] +name = "crc-catalog" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19d374276b40fb8bbdee95aef7c7fa6b5316ec764510eb64b8dd0e2ed0d7e7f5" + +[[package]] +name = "crc32fast" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "crossbeam-channel" +version = "0.5.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "33480d6946193aa8033910124896ca395333cae7e2d1113d1fef6c3272217df2" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-deque" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613f8cc01fe9cf1a3eb3d7f488fd2fa8388403e97039e2f73692932e291a770d" +dependencies = [ + "crossbeam-epoch", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.9.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80" + +[[package]] +name = "crunchy" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" + +[[package]] +name = "crypto-common" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3" +dependencies = [ + "generic-array", + "typenum", +] + +[[package]] +name = "cssparser" +version = "0.27.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "754b69d351cdc2d8ee09ae203db831e005560fc6030da058f86ad60c92a9cb0a" +dependencies = [ + "cssparser-macros", + "dtoa-short", + "itoa 0.4.8", + "matches", + "phf 0.8.0", + "proc-macro2", + "quote", + "smallvec", + "syn 1.0.109", +] + +[[package]] +name = "cssparser-macros" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13b588ba4ac1a99f7f2964d24b3d896ddc6bf847ee3855dbd4366f058cfcd331" +dependencies = [ + "quote", + "syn 2.0.86", +] + +[[package]] +name = "ctor" +version = "0.2.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "edb49164822f3ee45b17acd4a208cfc1251410cf0cad9a833234c9890774dd9f" +dependencies = [ + "quote", + "syn 2.0.86", +] + +[[package]] +name = "darling" +version = "0.20.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6f63b86c8a8826a49b8c21f08a2d07338eec8d900540f8630dc76284be802989" +dependencies = [ + "darling_core", + "darling_macro", +] + +[[package]] +name = "darling_core" +version = "0.20.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95133861a8032aaea082871032f5815eb9e98cef03fa916ab4500513994df9e5" +dependencies = [ + "fnv", + "ident_case", + "proc-macro2", + "quote", + "strsim 0.11.1", + "syn 2.0.86", +] + +[[package]] +name = "darling_macro" +version = "0.20.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d336a2a514f6ccccaa3e09b02d41d35330c07ddf03a62165fcec10bb561c7806" +dependencies = [ + "darling_core", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "data-url" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c297a1c74b71ae29df00c3e22dd9534821d60eb9af5a0192823fa2acea70c2a" + +[[package]] +name = "db" +version = "0.1.0" +dependencies = [ + "anyhow", + "rusqlite", + "serde", + "serde_json", + "strum", + "strum_macros", + "tempfile", +] + +[[package]] +name = "deflate64" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da692b8d1080ea3045efaab14434d40468c3d8657e42abddfffca87b428f4c1b" + +[[package]] +name = "deranged" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4" +dependencies = [ + "powerfmt", + "serde", +] + +[[package]] +name = "derivative" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fcc3dd5e9e9c0b295d6e1e4d811fb6f157d5ffd784b8d202fc62eac8035a770b" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "derive_arbitrary" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67e77553c4162a157adbf834ebae5b415acbecbeafc7a74b0e886657506a7611" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "derive_more" +version = "0.99.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f33878137e4dafd7fa914ad4e259e18a4e8e532b9617a2d0150262bf53abfce" +dependencies = [ + "convert_case", + "proc-macro2", + "quote", + "rustc_version", + "syn 2.0.86", +] + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer", + "crypto-common", + "subtle", +] + +[[package]] +name = "dirs" +version = "4.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca3aa72a6f96ea37bbc5aa912f6788242832f75369bdfdadcb0e38423f100059" +dependencies = [ + "dirs-sys 0.3.7", +] + +[[package]] +name = "dirs" +version = "5.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44c45a9d03d6676652bcb5e724c7e988de1acad23a711b5217ab9cbecbec2225" +dependencies = [ + "dirs-sys 0.4.1", +] + +[[package]] +name = "dirs-next" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b98cf8ebf19c3d1b223e151f99a4f9f0690dca41414773390fc824184ac833e1" +dependencies = [ + "cfg-if", + "dirs-sys-next", +] + +[[package]] +name = "dirs-sys" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b1d1d91c932ef41c0f2663aa8b0ca0342d444d842c06914aa0a7e352d0bada6" +dependencies = [ + "libc", + "redox_users", + "winapi", +] + +[[package]] +name = "dirs-sys" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "520f05a5cbd335fae5a99ff7a6ab8627577660ee5cfd6a94a6a929b52ff0321c" +dependencies = [ + "libc", + "option-ext", + "redox_users", + "windows-sys 0.48.0", +] + +[[package]] +name = "dirs-sys-next" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ebda144c4fe02d1f7ea1a7d9641b6fc6b580adcfa024ae48797ecdeb6825b4d" +dependencies = [ + "libc", + "redox_users", + "winapi", +] + +[[package]] +name = "dispatch" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd0c93bb4b0c6d9b77f4435b0ae98c24d17f1c45b2ff844c6151a07256ca923b" + +[[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.86", +] + +[[package]] +name = "dlib" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "330c60081dcc4c72131f8eb70510f1ac07223e5d4163db481a04a0befcffa412" +dependencies = [ + "libloading 0.8.5", +] + +[[package]] +name = "dlopen2" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e1297103d2bbaea85724fcee6294c2d50b1081f9ad47d0f6f6f61eda65315a6" +dependencies = [ + "dlopen2_derive", + "libc", + "once_cell", + "winapi", +] + +[[package]] +name = "dlopen2_derive" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f2b99bf03862d7f545ebc28ddd33a665b50865f4dfd84031a393823879bd4c54" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "dlv-list" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "442039f5147480ba31067cb00ada1adae6892028e40e45fc5de7b7df6dcc1b5f" +dependencies = [ + "const-random", +] + +[[package]] +name = "downcast-rs" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75b325c5dbd37f80359721ad39aca5a29fb04c89279657cffdda8736d0c0b9d2" + +[[package]] +name = "dpi" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f25c0e292a7ca6d6498557ff1df68f32c99850012b6ea401cf8daf771f22ff53" +dependencies = [ + "serde", +] + +[[package]] +name = "dtoa" +version = "1.0.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dcbb2bf8e87535c23f7a8a321e364ce21462d0ff10cb6407820e8e96dfff6653" + +[[package]] +name = "dtoa-short" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cd1511a7b6a56299bd043a9c167a6d2bfb37bf84a6dfceaba651168adfb43c87" +dependencies = [ + "dtoa", +] + +[[package]] +name = "dunce" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "92773504d58c093f6de2459af4af33faa518c13451eb8f2b5698ed3d36e7c813" + +[[package]] +name = "dyn-clone" +version = "1.0.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0d6ef0072f8a535281e4876be788938b528e9a1d43900b82c2569af7da799125" + +[[package]] +name = "either" +version = "1.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0" + +[[package]] +name = "embed-resource" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4e24052d7be71f0efb50c201557f6fe7d237cfd5a64fd5bcd7fd8fe32dbbffa" +dependencies = [ + "cc", + "memchr", + "rustc_version", + "toml 0.8.2", + "vswhom", + "winreg 0.52.0", +] + +[[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 = "endi" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3d8a32ae18130a3c84dd492d4215c3d913c3b07c6b63c2eb3eb7ff1101ab7bf" + +[[package]] +name = "enumflags2" +version = "0.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d232db7f5956f3f14313dc2f87985c58bd2c695ce124c8cdd984e08e15ac133d" +dependencies = [ + "enumflags2_derive", + "serde", +] + +[[package]] +name = "enumflags2_derive" +version = "0.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "de0d48a183585823424a4ce1aa132d174a6a81bd540895822eb4c8373a8e49e8" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "env_filter" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f2c92ceda6ceec50f43169f9ee8424fe2db276791afde7b2cd8bc084cb376ab" +dependencies = [ + "log", + "regex", +] + +[[package]] +name = "equivalent" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" + +[[package]] +name = "erased-serde" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24e2389d65ab4fab27dc2a5de7b191e1f6617d1f1c8855c0dc569c94a4cbb18d" +dependencies = [ + "serde", + "typeid", +] + +[[package]] +name = "errno" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "534c5cf6194dfab3db3242765c03bbe257cf92f22b38f6bc0c58d59108a820ba" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "error-code" +version = "3.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a5d9305ccc6942a704f4335694ecd3de2ea531b114ac2d51f5f843750787a92f" + +[[package]] +name = "event-listener" +version = "5.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6032be9bd27023a771701cc49f9f053c751055f71efb2e0ae5c15809093675ba" +dependencies = [ + "concurrent-queue", + "parking", + "pin-project-lite", +] + +[[package]] +name = "event-listener-strategy" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f214dc438f977e6d4e3500aaa277f5ad94ca83fbbd9b1a15713ce2344ccc5a1" +dependencies = [ + "event-listener", + "pin-project-lite", +] + +[[package]] +name = "exr" +version = "1.73.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f83197f59927b46c04a183a619b7c29df34e63e63c7869320862268c0ef687e0" +dependencies = [ + "bit_field", + "half", + "lebe", + "miniz_oxide", + "rayon-core", + "smallvec", + "zune-inflate", +] + +[[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.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8c02a5121d4ea3eb16a80748c74f5549a5665e4c21333c6098f283870fbdea6" + +[[package]] +name = "fdeflate" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07c6f4c64c1d33a3111c4466f7365ebdcc37c5bd1ea0d62aae2e3d722aacbedb" +dependencies = [ + "simd-adler32", +] + +[[package]] +name = "fern" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9f0c14694cbd524c8720dd69b0e3179344f04ebb5f90f2e4a440c6ea3b2f1ee" +dependencies = [ + "colored", + "log", +] + +[[package]] +name = "field-offset" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38e2275cc4e4fc009b0669731a1e5ab7ebf11f469eaede2bab9309a5b4d6057f" +dependencies = [ + "memoffset", + "rustc_version", +] + +[[package]] +name = "filetime" +version = "0.2.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "35c0522e981e68cbfa8c3f978441a5f34b30b96e146b33cd3359176b50fe8586" +dependencies = [ + "cfg-if", + "libc", + "libredox", + "windows-sys 0.59.0", +] + +[[package]] +name = "fixedbitset" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ce7134b9999ecaf8bcd65542e436736ef32ddca1b3e06094cb6ec5755203b80" + +[[package]] +name = "flate2" +version = "1.0.34" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1b589b4dc103969ad3cf85c950899926ec64300a1a46d76c03a6072957036f0" +dependencies = [ + "crc32fast", + "libz-ng-sys", + "miniz_oxide", +] + +[[package]] +name = "fluent-uri" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17c704e9dbe1ddd863da1e6ff3567795087b1eb201ce80d8fa81162e1516500d" +dependencies = [ + "bitflags 1.3.2", +] + +[[package]] +name = "flume" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da0e4dd2a88388a1f4ccc7c9ce104604dab68d9f408dc34cd45823d5a9069095" +dependencies = [ + "futures-core", + "futures-sink", + "spin", +] + +[[package]] +name = "fnv" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" + +[[package]] +name = "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.86", +] + +[[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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e13624c2627564efccf4934284bdd98cbaa14e79b0b5a141218e507b3a823456" +dependencies = [ + "percent-encoding", +] + +[[package]] +name = "funty" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c" + +[[package]] +name = "futf" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df420e2e84819663797d1ec6544b13c5be84629e7bb00dc960d6917db2987843" +dependencies = [ + "mac", + "new_debug_unreachable", +] + +[[package]] +name = "futures" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "65bc07b1a8bc7c85c5f2e110c476c7389b4554ba72af57d8445ea63a576b0876" +dependencies = [ + "futures-channel", + "futures-core", + "futures-executor", + "futures-io", + "futures-sink", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-channel" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2dff15bf788c671c1934e366d07e30c1814a8ef514e1af724a602e8a2fbe1b10" +dependencies = [ + "futures-core", + "futures-sink", +] + +[[package]] +name = "futures-core" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05f29059c0c2090612e8d742178b0580d2dc940c837851ad723096f87af6663e" + +[[package]] +name = "futures-executor" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e28d1d997f585e54aebc3f97d39e72338912123a67330d723fdbb564d646c9f" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-io" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e5c1b78ca4aae1ac06c48a526a655760685149f0d465d21f37abfe57ce075c6" + +[[package]] +name = "futures-lite" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f1fa2f9765705486b33fd2acf1577f8ec449c2ba1f318ae5447697b7c08d210" +dependencies = [ + "fastrand", + "futures-core", + "futures-io", + "parking", + "pin-project-lite", +] + +[[package]] +name = "futures-macro" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "futures-sink" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e575fab7d1e0dcb8d0c7bcf9a63ee213816ab51902e6d244a95819acacf1d4f7" + +[[package]] +name = "futures-task" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f90f7dce0722e95104fcb095585910c0977252f286e354b5e3bd38902cd99988" + +[[package]] +name = "futures-util" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-macro", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "pin-utils", + "slab", +] + +[[package]] +name = "fxhash" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c31b6d751ae2c7f11320402d34e41349dd1016f8d5d45e48c4312bc8625af50c" +dependencies = [ + "byteorder", +] + +[[package]] +name = "gdk" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f5ba081bdef3b75ebcdbfc953699ed2d7417d6bd853347a42a37d76406a33646" +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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31ff856cb3386dae1703a920f803abafcc580e9b5f711ca62ed1620c25b51ff2" +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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a90fbf5c033c65d93792192a49a8efb5bb1e640c419682a58bb96f5ae77f3d4a" +dependencies = [ + "gdk-sys", + "glib-sys", + "gobject-sys", + "libc", + "pkg-config", + "system-deps", +] + +[[package]] +name = "gdkx11" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db2ea8a4909d530f79921290389cbd7c34cb9d623bfe970eaae65ca5f9cd9cce" +dependencies = [ + "gdk", + "gdkx11-sys", + "gio", + "glib", + "libc", + "x11", +] + +[[package]] +name = "gdkx11-sys" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fee8f00f4ee46cad2939b8990f5c70c94ff882c3028f3cc5abf950fa4ab53043" +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 = "gethostname" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0176e0459c2e4a1fe232f984bca6890e681076abb9934f6cea7c326f3fc47818" +dependencies = [ + "libc", + "windows-targets 0.48.5", +] + +[[package]] +name = "gethostname" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc3655aa6818d65bc620d6911f05aa7b6aeb596291e1e9f79e52df85583d1e30" +dependencies = [ + "rustix", + "windows-targets 0.52.6", +] + +[[package]] +name = "getrandom" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" +dependencies = [ + "cfg-if", + "libc", + "wasi 0.9.0+wasi-snapshot-preview1", +] + +[[package]] +name = "getrandom" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7" +dependencies = [ + "cfg-if", + "libc", + "wasi 0.11.0+wasi-snapshot-preview1", +] + +[[package]] +name = "gif" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3fb2d69b19215e18bb912fa30f7ce15846e301408695e44e0ef719f1da9e19f2" +dependencies = [ + "color_quant", + "weezl", +] + +[[package]] +name = "gimli" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07e28edb80900c19c28f1072f2e8aeca7fa06b23cd4169cefe1af5aa3260783f" + +[[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", +] + +[[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.6.0", + "futures-channel", + "futures-core", + "futures-executor", + "futures-task", + "futures-util", + "gio-sys", + "glib-macros", + "glib-sys", + "gobject-sys", + "libc", + "memchr", + "once_cell", + "smallvec", + "thiserror", +] + +[[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.86", +] + +[[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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b" + +[[package]] +name = "global-hotkey" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b00d88f1be7bf4cd2e61623ce08e84be2dfa4eab458e5d632d3dab95f16c1f64" +dependencies = [ + "crossbeam-channel", + "keyboard-types", + "objc2", + "objc2-app-kit", + "once_cell", + "serde", + "thiserror", + "windows-sys 0.59.0", + "x11-dl", +] + +[[package]] +name = "globset" +version = "0.4.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15f1ce686646e7f1e19bf7d5533fe443a45dbfb990e00629110797578b42fb19" +dependencies = [ + "aho-corasick", + "bstr", + "log", + "regex-automata", + "regex-syntax", +] + +[[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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93c4f5e0e20b60e10631a5f06da7fe3dda744b05ad0ea71fee2f47adf865890c" +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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "771437bf1de2c1c0b496c11505bdf748e26066bbe942dfc8f614c9460f6d7722" +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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6063efb63db582968fb7df72e1ae68aa6360dcfb0a75143f34fc7d616bad75e" +dependencies = [ + "proc-macro-crate 1.3.1", + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "h2" +version = "0.3.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81fe527a889e1532da5c525686d96d4c2e74cdd345badf8dfef9f6b39dd5f5e8" +dependencies = [ + "bytes", + "fnv", + "futures-core", + "futures-sink", + "futures-util", + "http 0.2.12", + "indexmap 2.6.0", + "slab", + "tokio", + "tokio-util", + "tracing", +] + +[[package]] +name = "h2" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "524e8ac6999421f49a846c2d4411f337e53497d8ec55d67753beffa43c5d9205" +dependencies = [ + "atomic-waker", + "bytes", + "fnv", + "futures-core", + "futures-sink", + "http 1.1.0", + "indexmap 2.6.0", + "slab", + "tokio", + "tokio-util", + "tracing", +] + +[[package]] +name = "half" +version = "2.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6dd08c532ae367adf81c312a4580bc67f1d0fe8bc9c460520283f4c0ff277888" +dependencies = [ + "cfg-if", + "crunchy", +] + +[[package]] +name = "hashbrown" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" +dependencies = [ + "ahash 0.7.8", +] + +[[package]] +name = "hashbrown" +version = "0.14.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" +dependencies = [ + "ahash 0.8.11", +] + +[[package]] +name = "hashbrown" +version = "0.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e087f84d4f86bf4b218b927129862374b72199ae7d8657835f1e89000eea4fb" + +[[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 = "hermit-abi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024" + +[[package]] +name = "hermit-abi" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fbf6a919d6cf397374f7dfeeea91d974c7c0a7221d0d0f4f20d859d329e53fcc" + +[[package]] +name = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "hmac" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e" +dependencies = [ + "digest", +] + +[[package]] +name = "html5ever" +version = "0.26.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bea68cab48b8459f17cf1c944c67ddc572d272d9f2b274140f223ecb1da4a3b7" +dependencies = [ + "log", + "mac", + "markup5ever", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "http" +version = "0.2.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "601cbb57e577e2f5ef5be8e7b83f0f63994f25aa94d673e54a92d5c516d101f1" +dependencies = [ + "bytes", + "fnv", + "itoa 1.0.11", +] + +[[package]] +name = "http" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21b9ddb458710bc376481b842f5da65cdf31522de232c1ca8146abce2a358258" +dependencies = [ + "bytes", + "fnv", + "itoa 1.0.11", +] + +[[package]] +name = "http-body" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ceab25649e9960c0311ea418d17bee82c0dcec1bd053b5f9a66e265a693bed2" +dependencies = [ + "bytes", + "http 0.2.12", + "pin-project-lite", +] + +[[package]] +name = "http-body" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" +dependencies = [ + "bytes", + "http 1.1.0", +] + +[[package]] +name = "http-body-util" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" +dependencies = [ + "bytes", + "futures-util", + "http 1.1.0", + "http-body 1.0.1", + "pin-project-lite", +] + +[[package]] +name = "http-range-header" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "add0ab9360ddbd88cfeb3bd9574a1d85cfdfa14db10b3e21d3700dbc4328758f" + +[[package]] +name = "httparse" +version = "1.9.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7d71d3574edd2771538b901e6549113b4006ece66150fb69c0fb6d9a2adae946" + +[[package]] +name = "httpdate" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9" + +[[package]] +name = "hyper" +version = "0.14.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8c08302e8fa335b151b788c775ff56e7a03ae64ff85c548ee820fecb70356e85" +dependencies = [ + "bytes", + "futures-channel", + "futures-core", + "futures-util", + "h2 0.3.26", + "http 0.2.12", + "http-body 0.4.6", + "httparse", + "httpdate", + "itoa 1.0.11", + "pin-project-lite", + "socket2", + "tokio", + "tower-service", + "tracing", + "want", +] + +[[package]] +name = "hyper" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbbff0a806a4728c99295b254c8838933b5b082d75e3cb70c8dab21fdfbcfa9a" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "h2 0.4.6", + "http 1.1.0", + "http-body 1.0.1", + "httparse", + "itoa 1.0.11", + "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-rustls" +version = "0.27.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08afdbb5c31130e3034af566421053ab03787c640246a446327f550d11bcb333" +dependencies = [ + "futures-util", + "http 1.1.0", + "hyper 1.5.0", + "hyper-util", + "rustls 0.23.16", + "rustls-pki-types", + "tokio", + "tokio-rustls 0.26.0", + "tower-service", + "webpki-roots", +] + +[[package]] +name = "hyper-timeout" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbb958482e8c7be4bc3cf272a766a2b0bf1a6755e7a6ae777f017a31d11b13b1" +dependencies = [ + "hyper 0.14.31", + "pin-project-lite", + "tokio", + "tokio-io-timeout", +] + +[[package]] +name = "hyper-tls" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d6183ddfa99b85da61a140bea0efc93fdf56ceaa041b37d553518030827f9905" +dependencies = [ + "bytes", + "hyper 0.14.31", + "native-tls", + "tokio", + "tokio-native-tls", +] + +[[package]] +name = "hyper-util" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df2dcfbe0677734ab2f3ffa7fa7bfd4706bfdc1ef393f2ee30184aed67e631b4" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http 1.1.0", + "http-body 1.0.1", + "hyper 1.5.0", + "pin-project-lite", + "socket2", + "tokio", + "tower-service", + "tracing", +] + +[[package]] +name = "iana-time-zone" +version = "0.1.61" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "235e081f3925a06703c2d0117ea8b91f042756fd6e7a6e5d901e8ca1a996b220" +dependencies = [ + "android_system_properties", + "core-foundation-sys", + "iana-time-zone-haiku", + "js-sys", + "wasm-bindgen", + "windows-core 0.52.0", +] + +[[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.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3804960be0bb5e4edb1e1ad67afd321a9ecfd875c3e65c099468fd2717d7cae" +dependencies = [ + "byteorder", + "png", +] + +[[package]] +name = "ident_case" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" + +[[package]] +name = "idna" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e14ddfc70884202db2244c223200c204c2bda1bc6e0998d11b5e024d657209e6" +dependencies = [ + "unicode-bidi", + "unicode-normalization", +] + +[[package]] +name = "idna" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +dependencies = [ + "unicode-bidi", + "unicode-normalization", +] + +[[package]] +name = "if-addrs" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cabb0019d51a643781ff15c9c8a3e5dedc365c47211270f4e8f82812fedd8f0a" +dependencies = [ + "libc", + "windows-sys 0.48.0", +] + +[[package]] +name = "if-addrs" +version = "0.13.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a78a89907582615b19f6f0da1af18abf6ff08be259395669b834b057a7ee92d8" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "ignore" +version = "0.4.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6d89fd380afde86567dfba715db065673989d6253f42b88179abd3eae47bda4b" +dependencies = [ + "crossbeam-deque", + "globset", + "log", + "memchr", + "regex-automata", + "same-file", + "walkdir", + "winapi-util", +] + +[[package]] +name = "image" +version = "0.25.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bc144d44a31d753b02ce64093d532f55ff8dc4ebf2ffb8a63c0dda691385acae" +dependencies = [ + "bytemuck", + "byteorder-lite", + "color_quant", + "exr", + "gif", + "image-webp", + "num-traits", + "png", + "qoi", + "ravif", + "rayon", + "rgb", + "tiff", + "zune-core", + "zune-jpeg", +] + +[[package]] +name = "image-webp" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e031e8e3d94711a9ccb5d6ea357439ef3dcbed361798bd4071dc4d9793fbe22f" +dependencies = [ + "byteorder-lite", + "quick-error", +] + +[[package]] +name = "imgref" +version = "1.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d0263a3d970d5c054ed9312c0057b4f3bde9c0b33836d3637361d4a9e6e7a408" + +[[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.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "707907fe3c25f5424cce2cb7e1cbcafee6bdbe735ca90ef77c29e84591e5b9da" +dependencies = [ + "equivalent", + "hashbrown 0.15.0", + "serde", +] + +[[package]] +name = "infer" +version = "0.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bc150e5ce2330295b8616ce0e3f53250e53af31759a9dbedad1621ba29151847" +dependencies = [ + "cfb", +] + +[[package]] +name = "ini" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0a9271a5dfd4228fa56a78d7508a35c321639cc71f783bb7a5723552add87bce" +dependencies = [ + "configparser", +] + +[[package]] +name = "inout" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a0c10553d664a4d0bcff9f4215d0aac67a639cc68ef660840afe309b807bc9f5" +dependencies = [ + "generic-array", +] + +[[package]] +name = "instant" +version = "0.1.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0242819d153cba4b4b05a5a8f2a7e9bbf97b6055b2a002b395c96b5ff3c0222" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "interpolate_name" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c34819042dc3d3971c46c2190835914dfbe0c3c13f61449b2997f4e9722dfa60" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "ipnet" +version = "2.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ddc24109865250148c2e0f3d25d4f0f479571723792d3802153c60922a4fb708" + +[[package]] +name = "ipnetwork" +version = "0.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf466541e9d546596ee94f9f69590f89473455f88372423e0008fc1a7daf100e" +dependencies = [ + "serde", +] + +[[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-terminal" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "261f68e344040fbd0edea105bef17c66edf46f984ddb1115b775ce31be948f4b" +dependencies = [ + "hermit-abi 0.4.0", + "libc", + "windows-sys 0.52.0", +] + +[[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 = "is_terminal_polyfill" +version = "1.70.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf" + +[[package]] +name = "itertools" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba291022dbbd398a455acf126c1e341954079855bc60dfdda641363bd6922569" +dependencies = [ + "either", +] + +[[package]] +name = "itoa" +version = "0.4.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b71991ff56294aa922b450139ee08b3bfc70982c6b2c7562771375cf73542dd4" + +[[package]] +name = "itoa" +version = "1.0.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" + +[[package]] +name = "javascriptcore-rs" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca5671e9ffce8ffba57afc24070e906da7fc4b1ba66f2cabebf61bf2ea257fcc" +dependencies = [ + "bitflags 1.3.2", + "glib", + "javascriptcore-rs-sys", +] + +[[package]] +name = "javascriptcore-rs-sys" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af1be78d14ffa4b75b66df31840478fef72b51f8c2465d4ca7c194da9f7a5124" +dependencies = [ + "glib-sys", + "gobject-sys", + "libc", + "system-deps", +] + +[[package]] +name = "jni" +version = "0.21.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1a87aa2bb7d2af34197c04845522473242e1aa17c12f4935d5856491a7fb8c97" +dependencies = [ + "cesu8", + "cfg-if", + "combine", + "jni-sys", + "log", + "thiserror", + "walkdir", + "windows-sys 0.45.0", +] + +[[package]] +name = "jni-sys" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8eaf4bc02d17cbdd7ff4c7438cafcdf7fb9a4613313ad11b4f8fefe7d3fa0130" + +[[package]] +name = "jobserver" +version = "0.1.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48d1dbcbbeb6a7fec7e059840aa538bd62aaccf972c7346c4d9d2059312853d0" +dependencies = [ + "libc", +] + +[[package]] +name = "jpeg-decoder" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f5d4a7da358eff58addd2877a45865158f0d78c911d43a5784ceb7bbf52833b0" + +[[package]] +name = "js-sys" +version = "0.3.72" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a88f1bda2bd75b0452a14784937d796722fdebfe50df998aeb3f0b7603019a9" +dependencies = [ + "wasm-bindgen", +] + +[[package]] +name = "json-patch" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b1fb8864823fad91877e6caea0baca82e49e8db50f8e5c9f9a453e27d3330fc" +dependencies = [ + "jsonptr 0.4.7", + "serde", + "serde_json", + "thiserror", +] + +[[package]] +name = "json-patch" +version = "3.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "863726d7afb6bc2590eeff7135d923545e5e964f004c2ccf8716c25e70a86f08" +dependencies = [ + "jsonptr 0.6.3", + "serde", + "serde_json", + "thiserror", +] + +[[package]] +name = "jsonptr" +version = "0.4.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1c6e529149475ca0b2820835d3dce8fcc41c6b943ca608d32f35b449255e4627" +dependencies = [ + "fluent-uri", + "serde", + "serde_json", +] + +[[package]] +name = "jsonptr" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5dea2b27dd239b2556ed7a25ba842fe47fd602e7fc7433c2a8d6106d4d9edd70" +dependencies = [ + "serde", + "serde_json", +] + +[[package]] +name = "keyboard-types" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b750dcadc39a09dbadd74e118f6dd6598df77fa01df0cfcdc52c28dece74528a" +dependencies = [ + "bitflags 2.6.0", + "serde", + "unicode-segmentation", +] + +[[package]] +name = "kuchikiki" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f29e4755b7b995046f510a7520c42b2fed58b77bd94d5a87a8eb43d2fd126da8" +dependencies = [ + "cssparser", + "html5ever", + "indexmap 1.9.3", + "matches", + "selectors", +] + +[[package]] +name = "kunkun" +version = "0.0.0" +dependencies = [ + "anyhow", + "chrono", + "cocoa 0.24.1", + "log", + "mac-security-rs", + "mdns-sd", + "objc", + "serde", + "serde_json", + "tauri", + "tauri-build", + "tauri-plugin-cli", + "tauri-plugin-clipboard", + "tauri-plugin-deep-link", + "tauri-plugin-dialog", + "tauri-plugin-fs", + "tauri-plugin-global-shortcut", + "tauri-plugin-http", + "tauri-plugin-jarvis", + "tauri-plugin-log", + "tauri-plugin-network", + "tauri-plugin-notification", + "tauri-plugin-os", + "tauri-plugin-process", + "tauri-plugin-shell", + "tauri-plugin-shellx", + "tauri-plugin-single-instance", + "tauri-plugin-store", + "tauri-plugin-system-info", + "tauri-plugin-updater", + "tauri-plugin-upload", + "tokio", + "urlencoding", + "zip 2.2.0", +] + +[[package]] +name = "lazy_static" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe" + +[[package]] +name = "lazycell" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "830d08ce1d1d941e6b30645f1a0eb5643013d835ce3779a5fc208261dbe10f55" + +[[package]] +name = "lebe" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "03087c2bad5e1034e8cace5926dec053fb3790248370865f5117a7d0213354c8" + +[[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 0.7.4", + "once_cell", +] + +[[package]] +name = "libc" +version = "0.2.161" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e9489c2807c139ffd9c1794f4af0ebe86a828db53ecdc7fea2111d0fed085d1" + +[[package]] +name = "libfuzzer-sys" +version = "0.4.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a96cfd5557eb82f2b83fed4955246c988d331975a002961b07c81584d107e7f7" +dependencies = [ + "arbitrary", + "cc", + "once_cell", +] + +[[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 = "libloading" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4979f22fdb869068da03c9f7528f8297c6fd2606bc3a4affe42e6a823fdb8da4" +dependencies = [ + "cfg-if", + "windows-targets 0.52.6", +] + +[[package]] +name = "libredox" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c0ff37bd590ca25063e35af745c343cb7a0271906fb7b37e4813e8f79f00268d" +dependencies = [ + "bitflags 2.6.0", + "libc", + "redox_syscall", +] + +[[package]] +name = "libsqlite3-sys" +version = "0.28.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c10584274047cb335c23d3e61bcef8e323adae7c5c8c760540f73610177fc3f" +dependencies = [ + "cc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "libz-ng-sys" +version = "1.1.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f0f7295a34685977acb2e8cc8b08ee4a8dffd6cf278eeccddbe1ed55ba815d5" +dependencies = [ + "cmake", + "libc", +] + +[[package]] +name = "linux-raw-sys" +version = "0.4.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78b3ae25bc7c8c38cec158d1f2757ee79e9b3740fbc7ccf0e59e4b08d793fa89" + +[[package]] +name = "lnk" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e066ce29d4da51727b57c404c1270e3fa2a5ded0db1a4cb67c61f7a132421b2c" +dependencies = [ + "bitflags 1.3.2", + "byteorder", + "chrono", + "log", + "num-derive 0.3.3", + "num-traits", +] + +[[package]] +name = "localauthentication-rs" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b8efa96bad8c96e1e41778854c6ca043733dd93c8cf2f291c2d66ec34fafc26" +dependencies = [ + "swift-rs", +] + +[[package]] +name = "lock_api" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07af8b9cdd281b7915f413fa73f29ebd5d55d0d3f0155584dade1ff18cea1b17" +dependencies = [ + "autocfg", + "scopeguard", +] + +[[package]] +name = "lockfree-object-pool" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9374ef4228402d4b7e403e5838cb880d9ee663314b0a900d5a6aabf0c213552e" + +[[package]] +name = "log" +version = "0.4.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24" +dependencies = [ + "value-bag", +] + +[[package]] +name = "loop9" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fae87c125b03c1d2c0150c90365d7d6bcc53fb73a9acaef207d2d065860f062" +dependencies = [ + "imgref", +] + +[[package]] +name = "lzma-rs" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "297e814c836ae64db86b36cf2a557ba54368d03f6afcd7d947c266692f71115e" +dependencies = [ + "byteorder", + "crc", +] + +[[package]] +name = "mac" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c41e0c4fef86961ac6d6f8a82609f55f31b05e4fce149ac5710e439df7619ba4" + +[[package]] +name = "mac-notification-sys" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dce8f34f3717aa37177e723df6c1fc5fb02b2a1087374ea3fe0ea42316dc8f91" +dependencies = [ + "cc", + "dirs-next", + "objc-foundation", + "objc_id", + "time", +] + +[[package]] +name = "mac-security-rs" +version = "0.1.0" +dependencies = [ + "core-foundation 0.10.0", + "localauthentication-rs", + "objc", + "serde", +] + +[[package]] +name = "mach2" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19b955cdeb2a02b9117f121ce63aa52d08ade45de53e48fe6a38b39c10f6f709" +dependencies = [ + "libc", +] + +[[package]] +name = "malloc_buf" +version = "0.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62bb907fe88d54d8d9ce32a3cceab4218ed2f6b7d35617cafe9adf84e43919cb" +dependencies = [ + "libc", +] + +[[package]] +name = "markup5ever" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a2629bb1404f3d34c2e921f21fd34ba00b206124c81f65c50b43b6aaefeb016" +dependencies = [ + "log", + "phf 0.10.1", + "phf_codegen 0.10.0", + "string_cache", + "string_cache_codegen", + "tendril", +] + +[[package]] +name = "matches" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2532096657941c2fea9c289d370a250971c689d4f143798ff67113ec042024a5" + +[[package]] +name = "matchit" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e7465ac9959cc2b1404e8e2367b43684a6d13790fe23056cc8c6c5a6b7bcb94" + +[[package]] +name = "maybe-rayon" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ea1f30cedd69f0a2954655f7188c6a834246d2bcf1e315e2ac40c4b24dc9519" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "mdns-sd" +version = "0.11.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fe7c11a1eb3cfbfcf702d1601c1f5f4c102cdc8665b8a557783ef634741676e" +dependencies = [ + "flume", + "if-addrs 0.13.3", + "log", + "polling 2.8.0", + "socket2", +] + +[[package]] +name = "memchr" +version = "2.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3" + +[[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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a05b5d0594e0cb1ad8cee3373018d2b84e25905dc75b2468114cc9a8e86cfc20" + +[[package]] +name = "miniz_oxide" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2d80299ef12ff69b16a84bb182e3b9df68b5a91574d3d4fa6e41b65deec4df1" +dependencies = [ + "adler2", + "simd-adler32", +] + +[[package]] +name = "mio" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" +dependencies = [ + "hermit-abi 0.3.9", + "libc", + "wasi 0.11.0+wasi-snapshot-preview1", + "windows-sys 0.52.0", +] + +[[package]] +name = "muda" +version = "0.15.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b18047edf23933de40835403d4b9211ffd1dcc65c0eec569df38a1fb8aebd719" +dependencies = [ + "crossbeam-channel", + "dpi", + "gtk", + "keyboard-types", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "once_cell", + "png", + "serde", + "thiserror", + "windows-sys 0.59.0", +] + +[[package]] +name = "multimap" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "defc4c55412d89136f966bbb339008b474350e5e6e78d2714439c386b3137a03" + +[[package]] +name = "native-tls" +version = "0.2.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8614eb2c83d59d1c8cc974dd3f920198647674a0a035e1af1fa58707e317466" +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.6.0", + "jni-sys", + "log", + "ndk-sys", + "num_enum", + "raw-window-handle", + "thiserror", +] + +[[package]] +name = "ndk-context" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "27b02d87554356db9e9a873add8782d4ea6e3e58ea071a9adb9a2e8ddb884a8b" + +[[package]] +name = "ndk-sys" +version = "0.6.0+11769913" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee6cda3051665f1fb8d9e08fc35c96d5a244fb1be711a03b71118828afc9a873" +dependencies = [ + "jni-sys", +] + +[[package]] +name = "network-interface" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4a43439bf756eed340bdf8feba761e2d50c7d47175d87545cd5cbe4a137c4d1" +dependencies = [ + "cc", + "libc", + "serde", + "thiserror", + "winapi", +] + +[[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.27.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2eb04e9c688eff1c89d72b407f168cf79bb9e867a9d3323ed6c01519eb9cc053" +dependencies = [ + "bitflags 2.6.0", + "cfg-if", + "libc", + "memoffset", +] + +[[package]] +name = "nix" +version = "0.28.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ab2156c4fce2f8df6c499cc1c763e4394b7482525bf2a9701c9d79d215f519e4" +dependencies = [ + "bitflags 2.6.0", + "cfg-if", + "cfg_aliases 0.1.1", + "libc", +] + +[[package]] +name = "nodrop" +version = "0.1.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72ef4a56884ca558e5ddb05a1d1e7e1bfd9a68d9ed024c21704cc98872dae1bb" + +[[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 = "noop_proc_macro" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0676bb32a98c1a483ce53e500a81ad9c3d5b3f7c920c28c24e9cb0980d0b5bc8" + +[[package]] +name = "notify-rust" +version = "4.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5134a72dc570b178bff81b01e81ab14a6fcc015391ed4b3b14853090658cd3a3" +dependencies = [ + "log", + "mac-notification-sys", + "serde", + "tauri-winrt-notification", + "zbus", +] + +[[package]] +name = "ntapi" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8a3895c6391c39d7fe7ebc444a87eb2991b2a0bc718fdabd071eec617fc68e4" +dependencies = [ + "winapi", +] + +[[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.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "51d515d32fb182ee37cda2ccdcb92950d6a3c2893aa280e540671c2cd0f3b1d9" + +[[package]] +name = "num-derive" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "876a53fff98e03a936a674b29568b0e605f06b29372c2489ff4de23f1949743d" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "num-derive" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed3955f1a9c7c0c15e092f9c887db08b1fc683305fdf6eb6684f22555355e202" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[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-rational" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f83d14da390562dca69fc84082e73e548e1ad308d24accdedd2720017cb37824" +dependencies = [ + "num-bigint", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-traits" +version = "0.2.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +dependencies = [ + "autocfg", +] + +[[package]] +name = "num_cpus" +version = "1.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" +dependencies = [ + "hermit-abi 0.3.9", + "libc", +] + +[[package]] +name = "num_enum" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4e613fc340b2220f734a8595782c551f1250e969d87d3be1ae0579e8d4065179" +dependencies = [ + "num_enum_derive", +] + +[[package]] +name = "num_enum_derive" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af1844ef2428cc3e1cb900be36181049ef3d3193c63e43026cfe202983b27a56" +dependencies = [ + "proc-macro-crate 2.0.2", + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "num_threads" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c7398b9c8b70908f6371f47ed36737907c87c52af34c268fed0bf0ceb92ead9" +dependencies = [ + "libc", +] + +[[package]] +name = "objc" +version = "0.2.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "915b1b472bc21c53464d6c8461c9d3af805ba1ef837e1cac254428f4a77177b1" +dependencies = [ + "malloc_buf", +] + +[[package]] +name = "objc-foundation" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1add1b659e36c9607c7aab864a76c7a4c2760cd0cd2e120f3fb8b952c7e22bf9" +dependencies = [ + "block", + "objc", + "objc_id", +] + +[[package]] +name = "objc-sys" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cdb91bdd390c7ce1a8607f35f3ca7151b65afc0ff5ff3b34fa350f7d7c7e4310" +dependencies = [ + "cc", +] + +[[package]] +name = "objc2" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "46a785d4eeff09c14c487497c162e92766fbb3e4059a71840cecc03d9a50b804" +dependencies = [ + "objc-sys", + "objc2-encode", +] + +[[package]] +name = "objc2-app-kit" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e4e89ad9e3d7d297152b17d39ed92cd50ca8063a89a9fa569046d41568891eff" +dependencies = [ + "bitflags 2.6.0", + "block2", + "libc", + "objc2", + "objc2-core-data", + "objc2-core-image", + "objc2-foundation", + "objc2-quartz-core", +] + +[[package]] +name = "objc2-cloud-kit" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "74dd3b56391c7a0596a295029734d3c1c5e7e510a4cb30245f8221ccea96b009" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-core-location", + "objc2-foundation", +] + +[[package]] +name = "objc2-contacts" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a5ff520e9c33812fd374d8deecef01d4a840e7b41862d849513de77e44aa4889" +dependencies = [ + "block2", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-data" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "617fbf49e071c178c0b24c080767db52958f716d9eabdf0890523aeae54773ef" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-image" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55260963a527c99f1819c4f8e3b47fe04f9650694ef348ffd2227e8196d34c80" +dependencies = [ + "block2", + "objc2", + "objc2-foundation", + "objc2-metal", +] + +[[package]] +name = "objc2-core-location" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "000cfee34e683244f284252ee206a27953279d370e309649dc3ee317b37e5781" +dependencies = [ + "block2", + "objc2", + "objc2-contacts", + "objc2-foundation", +] + +[[package]] +name = "objc2-encode" +version = "4.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7891e71393cd1f227313c9379a26a584ff3d7e6e7159e988851f0934c993f0f8" + +[[package]] +name = "objc2-foundation" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ee638a5da3799329310ad4cfa62fbf045d5f56e3ef5ba4149e7452dcf89d5a8" +dependencies = [ + "bitflags 2.6.0", + "block2", + "dispatch", + "libc", + "objc2", +] + +[[package]] +name = "objc2-link-presentation" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1a1ae721c5e35be65f01a03b6d2ac13a54cb4fa70d8a5da293d7b0020261398" +dependencies = [ + "block2", + "objc2", + "objc2-app-kit", + "objc2-foundation", +] + +[[package]] +name = "objc2-metal" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd0cba1276f6023976a406a14ffa85e1fdd19df6b0f737b063b95f6c8c7aadd6" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-quartz-core" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e42bee7bff906b14b167da2bac5efe6b6a07e6f7c0a21a7308d40c960242dc7a" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-foundation", + "objc2-metal", +] + +[[package]] +name = "objc2-symbols" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0a684efe3dec1b305badae1a28f6555f6ddd3bb2c2267896782858d5a78404dc" +dependencies = [ + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-ui-kit" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8bb46798b20cd6b91cbd113524c490f1686f4c4e8f49502431415f3512e2b6f" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-cloud-kit", + "objc2-core-data", + "objc2-core-image", + "objc2-core-location", + "objc2-foundation", + "objc2-link-presentation", + "objc2-quartz-core", + "objc2-symbols", + "objc2-uniform-type-identifiers", + "objc2-user-notifications", +] + +[[package]] +name = "objc2-uniform-type-identifiers" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44fa5f9748dbfe1ca6c0b79ad20725a11eca7c2218bceb4b005cb1be26273bfe" +dependencies = [ + "block2", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-user-notifications" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76cfcbf642358e8689af64cee815d139339f3ed8ad05103ed5eaf73db8d84cb3" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-core-location", + "objc2-foundation", +] + +[[package]] +name = "objc2-web-kit" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68bc69301064cebefc6c4c90ce9cba69225239e4b8ff99d445a2b5563797da65" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-app-kit", + "objc2-foundation", +] + +[[package]] +name = "objc_id" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c92d4ddb4bd7b50d730c215ff871754d0da6b2178849f8a2a2ab69712d0c073b" +dependencies = [ + "objc", +] + +[[package]] +name = "object" +version = "0.36.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aedf0a2d09c573ed1d8d85b30c119153926a2b36dce0ab28322c09a117a4683e" +dependencies = [ + "memchr", +] + +[[package]] +name = "once_cell" +version = "1.20.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775" + +[[package]] +name = "open" +version = "5.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61a877bf6abd716642a53ef1b89fb498923a4afca5c754f9050b4d081c05c4b3" +dependencies = [ + "is-wsl", + "libc", + "pathdiff", +] + +[[package]] +name = "openssl" +version = "0.10.68" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6174bc48f102d208783c2c84bf931bb75927a617866870de8a4ea85597f871f5" +dependencies = [ + "bitflags 2.6.0", + "cfg-if", + "foreign-types 0.3.2", + "libc", + "once_cell", + "openssl-macros", + "openssl-sys", +] + +[[package]] +name = "openssl-macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "openssl-probe" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" + +[[package]] +name = "openssl-sys" +version = "0.9.104" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "45abf306cbf99debc8195b66b7346498d7b10c210de50418b5ccd7ceba08c741" +dependencies = [ + "cc", + "libc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "option-ext" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d" + +[[package]] +name = "ordered-multimap" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49203cdcae0030493bad186b28da2fa25645fa276a51b6fec8010d281e02ef79" +dependencies = [ + "dlv-list", + "hashbrown 0.14.5", +] + +[[package]] +name = "ordered-stream" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9aa2b01e1d916879f73a53d01d1d6cee68adbb31d6d9177a8cfce093cced1d50" +dependencies = [ + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "os_info" +version = "3.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ae99c7fa6dd38c7cafe1ec085e804f8f555a2f8659b0dbe03f1f9963a9b51092" +dependencies = [ + "log", + "serde", + "windows-sys 0.52.0", +] + +[[package]] +name = "os_pipe" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ffd2b0a5634335b135d5728d84c5e0fd726954b87111f7506a61c502280d982" +dependencies = [ + "libc", + "windows-sys 0.59.0", +] + +[[package]] +name = "pango" +version = "0.18.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ca27ec1eb0457ab26f3036ea52229edbdb74dee1edd29063f5b9b010e7ebee4" +dependencies = [ + "gio", + "glib", + "libc", + "once_cell", + "pango-sys", +] + +[[package]] +name = "pango-sys" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "436737e391a843e5933d6d9aa102cb126d501e815b83601365a948a518555dc5" +dependencies = [ + "glib-sys", + "gobject-sys", + "libc", + "system-deps", +] + +[[package]] +name = "parking" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba" + +[[package]] +name = "parking_lot" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27" +dependencies = [ + "lock_api", + "parking_lot_core", +] + +[[package]] +name = "parking_lot_core" +version = "0.9.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall", + "smallvec", + "windows-targets 0.52.6", +] + +[[package]] +name = "parselnk" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0088616e6efe53ab79907b9313f4743eb3f8a16eb1e0014af810164808906dc3" +dependencies = [ + "bitflags 1.3.2", + "byteorder", + "chrono", + "thiserror", + "widestring", +] + +[[package]] +name = "paste" +version = "1.0.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a" + +[[package]] +name = "pathdiff" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d61c5ce1153ab5b689d0c074c4e7fc613e942dfb7dd9eea5ab202d2ad91fe361" + +[[package]] +name = "pbkdf2" +version = "0.12.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8ed6a7761f76e3b9f92dfb0a60a6a6477c61024b775147ff0973a02653abaf2" +dependencies = [ + "digest", + "hmac", +] + +[[package]] +name = "percent-encoding" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" + +[[package]] +name = "petgraph" +version = "0.6.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b4c5cc86750666a3ed20bdaf5ca2a0344f9c67674cae0515bec2da16fbaa47db" +dependencies = [ + "fixedbitset", + "indexmap 2.6.0", +] + +[[package]] +name = "phf" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3dfb61232e34fcb633f43d12c58f83c1df82962dcdfa565a4e866ffc17dafe12" +dependencies = [ + "phf_macros 0.8.0", + "phf_shared 0.8.0", + "proc-macro-hack", +] + +[[package]] +name = "phf" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fabbf1ead8a5bcbc20f5f8b939ee3f5b0f6f281b6ad3468b84656b658b455259" +dependencies = [ + "phf_shared 0.10.0", +] + +[[package]] +name = "phf" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ade2d8b8f33c7333b51bcf0428d37e217e9f32192ae4772156f65063b8ce03dc" +dependencies = [ + "phf_macros 0.11.2", + "phf_shared 0.11.2", +] + +[[package]] +name = "phf_codegen" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cbffee61585b0411840d3ece935cce9cb6321f01c45477d30066498cd5e1a815" +dependencies = [ + "phf_generator 0.8.0", + "phf_shared 0.8.0", +] + +[[package]] +name = "phf_codegen" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4fb1c3a8bc4dd4e5cfce29b44ffc14bedd2ee294559a294e2a4d4c9e9a6a13cd" +dependencies = [ + "phf_generator 0.10.0", + "phf_shared 0.10.0", +] + +[[package]] +name = "phf_generator" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17367f0cc86f2d25802b2c26ee58a7b23faeccf78a396094c13dced0d0182526" +dependencies = [ + "phf_shared 0.8.0", + "rand 0.7.3", +] + +[[package]] +name = "phf_generator" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d5285893bb5eb82e6aaf5d59ee909a06a16737a8970984dd7746ba9283498d6" +dependencies = [ + "phf_shared 0.10.0", + "rand 0.8.5", +] + +[[package]] +name = "phf_generator" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48e4cc64c2ad9ebe670cb8fd69dd50ae301650392e81c05f9bfcb2d5bdbc24b0" +dependencies = [ + "phf_shared 0.11.2", + "rand 0.8.5", +] + +[[package]] +name = "phf_macros" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f6fde18ff429ffc8fe78e2bf7f8b7a5a5a6e2a8b58bc5a9ac69198bbda9189c" +dependencies = [ + "phf_generator 0.8.0", + "phf_shared 0.8.0", + "proc-macro-hack", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "phf_macros" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3444646e286606587e49f3bcf1679b8cef1dc2c5ecc29ddacaffc305180d464b" +dependencies = [ + "phf_generator 0.11.2", + "phf_shared 0.11.2", + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "phf_shared" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c00cf8b9eafe68dde5e9eaa2cef8ee84a9336a47d566ec55ca16589633b65af7" +dependencies = [ + "siphasher", +] + +[[package]] +name = "phf_shared" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6796ad771acdc0123d2a88dc428b5e38ef24456743ddb1744ed628f9815c096" +dependencies = [ + "siphasher", +] + +[[package]] +name = "phf_shared" +version = "0.11.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90fcb95eef784c2ac79119d1dd819e162b5da872ce6f3c3abe1e8ca1c082f72b" +dependencies = [ + "siphasher", +] + +[[package]] +name = "pin-project" +version = "1.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be57f64e946e500c8ee36ef6331845d40a93055567ec57e8fae13efd33759b95" +dependencies = [ + "pin-project-internal", +] + +[[package]] +name = "pin-project-internal" +version = "1.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c0f5fad0874fc7abcd4d750e76917eaebbecaa2c20bde22e1dbeeba8beb758c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "pin-project-lite" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "915a1e146535de9163f3987b8944ed8cf49a18bb0056bcebcdcece385cece4ff" + +[[package]] +name = "pin-utils" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" + +[[package]] +name = "piper" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96c8c490f422ef9a4efd2cb5b42b76c8613d7e7dfc1caf667b8a3350a5acc066" +dependencies = [ + "atomic-waker", + "fastrand", + "futures-io", +] + +[[package]] +name = "pkg-config" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2" + +[[package]] +name = "plist" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42cf17e9a1800f5f396bc67d193dc9411b59012a5876445ef450d449881e1016" +dependencies = [ + "base64 0.22.1", + "indexmap 2.6.0", + "quick-xml 0.32.0", + "serde", + "time", +] + +[[package]] +name = "png" +version = "0.17.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52f9d46a34a05a6a57566bc2bfae066ef07585a6e3fa30fbbdff5936380623f0" +dependencies = [ + "bitflags 1.3.2", + "crc32fast", + "fdeflate", + "flate2", + "miniz_oxide", +] + +[[package]] +name = "polling" +version = "2.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4b2d323e8ca7996b3e23126511a523f7e62924d93ecd5ae73b333815b0eb3dce" +dependencies = [ + "autocfg", + "bitflags 1.3.2", + "cfg-if", + "concurrent-queue", + "libc", + "log", + "pin-project-lite", + "windows-sys 0.48.0", +] + +[[package]] +name = "polling" +version = "3.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc2790cd301dec6cd3b7a025e4815cf825724a51c98dccfe6a3e55f05ffb6511" +dependencies = [ + "cfg-if", + "concurrent-queue", + "hermit-abi 0.4.0", + "pin-project-lite", + "rustix", + "tracing", + "windows-sys 0.59.0", +] + +[[package]] +name = "powerfmt" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391" + +[[package]] +name = "ppv-lite86" +version = "0.2.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04" +dependencies = [ + "zerocopy", +] + +[[package]] +name = "precomputed-hash" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "925383efa346730478fb4838dbe9137d2a47675ad789c546d150a6e1dd4ab31c" + +[[package]] +name = "prettyplease" +version = "0.2.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "64d1ec885c64d0457d564db4ec299b2dae3f9c02808b8ad9c3a089c591b18033" +dependencies = [ + "proc-macro2", + "syn 2.0.86", +] + +[[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", + "toml_edit 0.20.2", +] + +[[package]] +name = "proc-macro-error" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c" +dependencies = [ + "proc-macro-error-attr", + "proc-macro2", + "quote", + "syn 1.0.109", + "version_check", +] + +[[package]] +name = "proc-macro-error-attr" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869" +dependencies = [ + "proc-macro2", + "quote", + "version_check", +] + +[[package]] +name = "proc-macro-hack" +version = "0.5.20+deprecated" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc375e1527247fe1a97d8b7156678dfe7c1af2fc075c9a4db3690ecd2a148068" + +[[package]] +name = "proc-macro2" +version = "1.0.89" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f139b0662de085916d1fb67d2b4169d1addddda1919e696f3252b740b629986e" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "profiling" +version = "1.0.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "afbdc74edc00b6f6a218ca6a5364d6226a259d4b8ea1af4a0ea063f27e179f4d" +dependencies = [ + "profiling-procmacros", +] + +[[package]] +name = "profiling-procmacros" +version = "1.0.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a65f2e60fbf1063868558d69c6beacf412dc755f9fc020f514b7955fc914fe30" +dependencies = [ + "quote", + "syn 2.0.86", +] + +[[package]] +name = "prost" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "deb1435c188b76130da55f17a466d252ff7b1418b2ad3e037d127b94e3411f29" +dependencies = [ + "bytes", + "prost-derive", +] + +[[package]] +name = "prost-build" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22505a5c94da8e3b7c2996394d1c933236c4d743e81a410bcca4e6989fc066a4" +dependencies = [ + "bytes", + "heck 0.5.0", + "itertools", + "log", + "multimap", + "once_cell", + "petgraph", + "prettyplease", + "prost", + "prost-types", + "regex", + "syn 2.0.86", + "tempfile", +] + +[[package]] +name = "prost-derive" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81bddcdb20abf9501610992b6759a4c888aef7d1a7247ef75e2404275ac24af1" +dependencies = [ + "anyhow", + "itertools", + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "prost-types" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9091c90b0a32608e984ff2fa4091273cbdd755d54935c51d520887f4a1dbd5b0" +dependencies = [ + "prost", +] + +[[package]] +name = "psl-types" +version = "2.0.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "33cb294fe86a74cbcf50d4445b37da762029549ebeea341421c7c70370f86cac" + +[[package]] +name = "ptr_meta" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0738ccf7ea06b608c10564b31debd4f5bc5e197fc8bfe088f68ae5ce81e7a4f1" +dependencies = [ + "ptr_meta_derive", +] + +[[package]] +name = "ptr_meta_derive" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "16b845dbfca988fa33db069c0e230574d15a3088f147a87b64c7589eb662c9ac" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "publicsuffix" +version = "2.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96a8c1bda5ae1af7f99a2962e49df150414a43d62404644d98dd5c3a93d07457" +dependencies = [ + "idna 0.3.0", + "psl-types", +] + +[[package]] +name = "qoi" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f6d64c71eb498fe9eae14ce4ec935c555749aef511cca85b5568910d6e48001" +dependencies = [ + "bytemuck", +] + +[[package]] +name = "quick-error" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a993555f31e5a609f617c12db6250dedcac1b0a85076912c436e6fc9b2c8e6a3" + +[[package]] +name = "quick-xml" +version = "0.31.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1004a344b30a54e2ee58d66a71b32d2db2feb0a31f9a2d302bf0536f15de2a33" +dependencies = [ + "memchr", +] + +[[package]] +name = "quick-xml" +version = "0.32.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d3a6e5838b60e0e8fa7a43f22ade549a37d61f8bdbe636d0d7816191de969c2" +dependencies = [ + "memchr", +] + +[[package]] +name = "quick-xml" +version = "0.36.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f7649a7b4df05aed9ea7ec6f628c67c9953a43869b8bc50929569b2999d443fe" +dependencies = [ + "memchr", +] + +[[package]] +name = "quinn" +version = "0.11.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8c7c5fdde3cdae7203427dc4f0a68fe0ed09833edc525a03456b153b79828684" +dependencies = [ + "bytes", + "pin-project-lite", + "quinn-proto", + "quinn-udp", + "rustc-hash", + "rustls 0.23.16", + "socket2", + "thiserror", + "tokio", + "tracing", +] + +[[package]] +name = "quinn-proto" +version = "0.11.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fadfaed2cd7f389d0161bb73eeb07b7b78f8691047a6f3e73caaeae55310a4a6" +dependencies = [ + "bytes", + "rand 0.8.5", + "ring", + "rustc-hash", + "rustls 0.23.16", + "slab", + "thiserror", + "tinyvec", + "tracing", +] + +[[package]] +name = "quinn-udp" +version = "0.5.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e346e016eacfff12233c243718197ca12f148c84e1e84268a896699b41c71780" +dependencies = [ + "cfg_aliases 0.2.1", + "libc", + "once_cell", + "socket2", + "tracing", + "windows-sys 0.59.0", +] + +[[package]] +name = "quote" +version = "1.0.37" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5b9d34b8991d19d98081b46eacdd8eb58c6f2b201139f7c5f643cc155a633af" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "radium" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc33ff2d4973d518d823d61aa239014831e521c75da58e3df4840d3f47749d09" + +[[package]] +name = "rand" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03" +dependencies = [ + "getrandom 0.1.16", + "libc", + "rand_chacha 0.2.2", + "rand_core 0.5.1", + "rand_hc", + "rand_pcg", +] + +[[package]] +name = "rand" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" +dependencies = [ + "libc", + "rand_chacha 0.3.1", + "rand_core 0.6.4", +] + +[[package]] +name = "rand_chacha" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402" +dependencies = [ + "ppv-lite86", + "rand_core 0.5.1", +] + +[[package]] +name = "rand_chacha" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" +dependencies = [ + "ppv-lite86", + "rand_core 0.6.4", +] + +[[package]] +name = "rand_core" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19" +dependencies = [ + "getrandom 0.1.16", +] + +[[package]] +name = "rand_core" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" +dependencies = [ + "getrandom 0.2.15", +] + +[[package]] +name = "rand_hc" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c" +dependencies = [ + "rand_core 0.5.1", +] + +[[package]] +name = "rand_pcg" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "16abd0c1b639e9eb4d7c50c0b8100b0d0f849be2349829c740fe8e6eb4816429" +dependencies = [ + "rand_core 0.5.1", +] + +[[package]] +name = "rav1e" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cd87ce80a7665b1cce111f8a16c1f3929f6547ce91ade6addf4ec86a8dda5ce9" +dependencies = [ + "arbitrary", + "arg_enum_proc_macro", + "arrayvec", + "av1-grain", + "bitstream-io", + "built", + "cfg-if", + "interpolate_name", + "itertools", + "libc", + "libfuzzer-sys", + "log", + "maybe-rayon", + "new_debug_unreachable", + "noop_proc_macro", + "num-derive 0.4.2", + "num-traits", + "once_cell", + "paste", + "profiling", + "rand 0.8.5", + "rand_chacha 0.3.1", + "simd_helpers", + "system-deps", + "thiserror", + "v_frame", + "wasm-bindgen", +] + +[[package]] +name = "ravif" +version = "0.11.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2413fd96bd0ea5cdeeb37eaf446a22e6ed7b981d792828721e74ded1980a45c6" +dependencies = [ + "avif-serialize", + "imgref", + "loop9", + "quick-error", + "rav1e", + "rgb", +] + +[[package]] +name = "raw-window-handle" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20675572f6f24e9e76ef639bc5552774ed45f1c30e2951e1e99c59888861c539" + +[[package]] +name = "rayon" +version = "1.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b418a60154510ca1a002a752ca9714984e21e4241e804d32555251faf8b78ffa" +dependencies = [ + "either", + "rayon-core", +] + +[[package]] +name = "rayon-core" +version = "1.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1465873a3dfdaa8ae7cb14b4383657caab0b3e8a0aa9ae8e04b044854c8dfce2" +dependencies = [ + "crossbeam-deque", + "crossbeam-utils", +] + +[[package]] +name = "read-progress-stream" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6435842fc2fea44b528719eb8c32203bbc1bb2f5b619fbe0c0a3d8350fd8d2a8" +dependencies = [ + "bytes", + "futures", + "pin-project-lite", +] + +[[package]] +name = "redox_syscall" +version = "0.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b6dfecf2c74bce2466cabf93f6664d6998a69eb21e39f4207930065b27b771f" +dependencies = [ + "bitflags 2.6.0", +] + +[[package]] +name = "redox_users" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba009ff324d1fc1b900bd1fdb31564febe58a8ccc8a6fdbb93b543d33b13ca43" +dependencies = [ + "getrandom 0.2.15", + "libredox", + "thiserror", +] + +[[package]] +name = "regex" +version = "1.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191" +dependencies = [ + "aho-corasick", + "memchr", + "regex-automata", + "regex-syntax", +] + +[[package]] +name = "regex-automata" +version = "0.4.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "368758f23274712b504848e9d5a6f010445cc8b87a7cdb4d7cbee666c1288da3" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", +] + +[[package]] +name = "regex-syntax" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c" + +[[package]] +name = "rend" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "71fe3824f5629716b1589be05dacd749f6aa084c87e00e016714a8cdfccc997c" +dependencies = [ + "bytecheck", +] + +[[package]] +name = "reqwest" +version = "0.11.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd67538700a17451e7cba03ac727fb961abb7607553461627b97de0b89cf4a62" +dependencies = [ + "base64 0.21.7", + "bytes", + "encoding_rs", + "futures-core", + "futures-util", + "h2 0.3.26", + "http 0.2.12", + "http-body 0.4.6", + "hyper 0.14.31", + "hyper-tls", + "ipnet", + "js-sys", + "log", + "mime", + "native-tls", + "once_cell", + "percent-encoding", + "pin-project-lite", + "rustls-pemfile 1.0.4", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper 0.1.2", + "system-configuration 0.5.1", + "tokio", + "tokio-native-tls", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "winreg 0.50.0", +] + +[[package]] +name = "reqwest" +version = "0.12.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a77c62af46e79de0a562e1a9849205ffcb7fc1238876e9bd743357570e04046f" +dependencies = [ + "base64 0.22.1", + "bytes", + "cookie", + "cookie_store", + "encoding_rs", + "futures-core", + "futures-util", + "h2 0.4.6", + "http 1.1.0", + "http-body 1.0.1", + "http-body-util", + "hyper 1.5.0", + "hyper-rustls", + "hyper-util", + "ipnet", + "js-sys", + "log", + "mime", + "once_cell", + "percent-encoding", + "pin-project-lite", + "quinn", + "rustls 0.23.16", + "rustls-pemfile 2.2.0", + "rustls-pki-types", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper 1.0.1", + "system-configuration 0.6.1", + "tokio", + "tokio-rustls 0.26.0", + "tokio-util", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-streams", + "web-sys", + "webpki-roots", + "windows-registry 0.2.0", +] + +[[package]] +name = "rfd" +version = "0.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8af382a047821a08aa6bfc09ab0d80ff48d45d8726f7cd8e44891f7cb4a4278e" +dependencies = [ + "ashpd", + "block2", + "glib-sys", + "gobject-sys", + "gtk-sys", + "js-sys", + "log", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "raw-window-handle", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "windows-sys 0.48.0", +] + +[[package]] +name = "rgb" +version = "0.8.50" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "57397d16646700483b67d2dd6511d79318f9d057fdbd21a4066aeac8b41d310a" + +[[package]] +name = "ring" +version = "0.17.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c17fa4cb658e3583423e915b9f3acc01cceaee1860e33d59ebae66adc3a2dc0d" +dependencies = [ + "cc", + "cfg-if", + "getrandom 0.2.15", + "libc", + "spin", + "untrusted", + "windows-sys 0.52.0", +] + +[[package]] +name = "rkyv" +version = "0.7.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9008cd6385b9e161d8229e1f6549dd23c3d022f132a2ea37ac3a10ac4935779b" +dependencies = [ + "bitvec", + "bytecheck", + "bytes", + "hashbrown 0.12.3", + "ptr_meta", + "rend", + "rkyv_derive", + "seahash", + "tinyvec", + "uuid", +] + +[[package]] +name = "rkyv_derive" +version = "0.7.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "503d1d27590a2b0a3a4ca4c94755aa2875657196ecbf401a42eff41d7de532c0" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "rusqlite" +version = "0.31.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b838eba278d213a8beaf485bd313fd580ca4505a00d5871caeb1457c55322cae" +dependencies = [ + "bitflags 2.6.0", + "fallible-iterator", + "fallible-streaming-iterator", + "hashlink", + "libsqlite3-sys", + "smallvec", +] + +[[package]] +name = "rust-ini" +version = "0.21.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4e310ef0e1b6eeb79169a1171daf9abcb87a2e17c03bee2c4bb100b55c75409f" +dependencies = [ + "cfg-if", + "ordered-multimap", + "trim-in-place", +] + +[[package]] +name = "rust_decimal" +version = "1.36.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b082d80e3e3cc52b2ed634388d436fe1f4de6af5786cc2de9ba9737527bdf555" +dependencies = [ + "arrayvec", + "borsh", + "bytes", + "num-traits", + "rand 0.8.5", + "rkyv", + "serde", + "serde_json", +] + +[[package]] +name = "rust_search" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d27d7be20245d289c9dde663f06521de08663d73cbaefc45785aa65d02022378" +dependencies = [ + "dirs 4.0.0", + "ignore", + "num_cpus", + "regex", + "strsim 0.10.0", +] + +[[package]] +name = "rustc-demangle" +version = "0.1.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "719b953e2095829ee67db738b3bfa9fa368c94900df327b3f07fe6e794d2fe1f" + +[[package]] +name = "rustc-hash" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "583034fd73374156e66797ed8e5b0d5690409c9226b22d87cb7f19821c05d152" + +[[package]] +name = "rustc_version" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cfcb3a22ef46e85b45de6ee7e79d063319ebb6594faafcf1c225ea92ab6e9b92" +dependencies = [ + "semver", +] + +[[package]] +name = "rustix" +version = "0.38.38" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa260229e6538e52293eeb577aabd09945a09d6d9cc0fc550ed7529056c2e32a" +dependencies = [ + "bitflags 2.6.0", + "errno", + "libc", + "linux-raw-sys", + "windows-sys 0.52.0", +] + +[[package]] +name = "rustls" +version = "0.21.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f56a14d1f48b391359b22f731fd4bd7e43c97f3c50eee276f3aa09c94784d3e" +dependencies = [ + "log", + "ring", + "rustls-webpki 0.101.7", + "sct", +] + +[[package]] +name = "rustls" +version = "0.23.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eee87ff5d9b36712a58574e12e9f0ea80f915a5b0ac518d322b24a465617925e" +dependencies = [ + "once_cell", + "ring", + "rustls-pki-types", + "rustls-webpki 0.102.8", + "subtle", + "zeroize", +] + +[[package]] +name = "rustls-pemfile" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1c74cae0a4cf6ccbbf5f359f08efdf8ee7e1dc532573bf0db71968cb56b1448c" +dependencies = [ + "base64 0.21.7", +] + +[[package]] +name = "rustls-pemfile" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dce314e5fee3f39953d46bb63bb8a46d40c2f8fb7cc5a3b6cab2bde9721d6e50" +dependencies = [ + "rustls-pki-types", +] + +[[package]] +name = "rustls-pki-types" +version = "1.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "16f1201b3c9a7ee8039bcadc17b7e605e2945b27eee7631788c1bd2b0643674b" + +[[package]] +name = "rustls-webpki" +version = "0.101.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b6275d1ee7a1cd780b64aca7726599a1dbc893b1e64144529e55c3c2f745765" +dependencies = [ + "ring", + "untrusted", +] + +[[package]] +name = "rustls-webpki" +version = "0.102.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "64ca1bc8749bd4cf37b5ce386cc146580777b4e8572c7b97baf22c83f444bee9" +dependencies = [ + "ring", + "rustls-pki-types", + "untrusted", +] + +[[package]] +name = "rustversion" +version = "1.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e819f2bc632f285be6d7cd36e25940d45b2391dd6d9b939e79de557f7014248" + +[[package]] +name = "ryu" +version = "1.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f" + +[[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.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "01227be5826fa0690321a2ba6c5cd57a19cf3f6a09e76973b58e61de6ab9d1c1" +dependencies = [ + "windows-sys 0.59.0", +] + +[[package]] +name = "schemars" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09c024468a378b7e36765cd36702b7a90cc3cba11654f6685c8f233408e89e92" +dependencies = [ + "dyn-clone", + "indexmap 1.9.3", + "schemars_derive", + "serde", + "serde_json", + "url", + "uuid", +] + +[[package]] +name = "schemars_derive" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1eee588578aff73f856ab961cd2f79e36bc45d7ded33a7562adba4667aecc0e" +dependencies = [ + "proc-macro2", + "quote", + "serde_derive_internals", + "syn 2.0.86", +] + +[[package]] +name = "scoped-tls" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e1cf6437eb19a8f4a6cc0f7dca544973b0b78843adbfeb3683d1a94a0024a294" + +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + +[[package]] +name = "sct" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da046153aa2352493d6cb7da4b6e5c0c057d8a1d0a9aa8560baffdd945acd414" +dependencies = [ + "ring", + "untrusted", +] + +[[package]] +name = "seahash" +version = "4.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1c107b6f4780854c8b126e228ea8869f4d7b71260f962fefb57b996b8959ba6b" + +[[package]] +name = "security-framework" +version = "2.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "897b2245f0b511c87893af39b033e5ca9cce68824c4d7e7630b5a1d339658d02" +dependencies = [ + "bitflags 2.6.0", + "core-foundation 0.9.4", + "core-foundation-sys", + "libc", + "security-framework-sys", +] + +[[package]] +name = "security-framework-sys" +version = "2.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ea4a292869320c0272d7bc55a5a6aafaff59b4f63404a003887b679a2e05b4b6" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "selectors" +version = "0.22.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df320f1889ac4ba6bc0cdc9c9af7af4bd64bb927bccdf32d81140dc1f9be12fe" +dependencies = [ + "bitflags 1.3.2", + "cssparser", + "derive_more", + "fxhash", + "log", + "matches", + "phf 0.8.0", + "phf_codegen 0.8.0", + "precomputed-hash", + "servo_arc", + "smallvec", + "thin-slice", +] + +[[package]] +name = "semver" +version = "1.0.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61697e0a1c7e512e84a621326239844a24d8207b4669b41bc18b32ea5cbf988b" +dependencies = [ + "serde", +] + +[[package]] +name = "serde" +version = "1.0.214" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f55c3193aca71c12ad7890f1785d2b73e1b9f63a0bbc353c08ef26fe03fc56b5" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde-untagged" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2676ba99bd82f75cae5cbd2c8eda6fa0b8760f18978ea840e980dd5567b5c5b6" +dependencies = [ + "erased-serde", + "serde", + "typeid", +] + +[[package]] +name = "serde_derive" +version = "1.0.214" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "de523f781f095e28fa605cdce0f8307e451cc0fd14e2eb4cd2e98a355b147766" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[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.86", +] + +[[package]] +name = "serde_json" +version = "1.0.132" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d726bfaff4b320266d395898905d0eba0345aae23b54aee3a737e260fd46db03" +dependencies = [ + "itoa 1.0.11", + "memchr", + "ryu", + "serde", +] + +[[package]] +name = "serde_path_to_error" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af99884400da37c88f5e9146b7f1fd0fbcae8f6eec4e9da38b67d05486f814a6" +dependencies = [ + "itoa 1.0.11", + "serde", +] + +[[package]] +name = "serde_repr" +version = "0.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c64451ba24fc7a6a2d60fc75dd9c83c90903b19028d4eff35e88fc1e86564e9" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "serde_spanned" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1" +dependencies = [ + "serde", +] + +[[package]] +name = "serde_urlencoded" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" +dependencies = [ + "form_urlencoded", + "itoa 1.0.11", + "ryu", + "serde", +] + +[[package]] +name = "serde_with" +version = "3.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e28bdad6db2b8340e449f7108f020b3b092e8583a9e3fb82713e1d4e71fe817" +dependencies = [ + "base64 0.22.1", + "chrono", + "hex", + "indexmap 1.9.3", + "indexmap 2.6.0", + "serde", + "serde_derive", + "serde_json", + "serde_with_macros", + "time", +] + +[[package]] +name = "serde_with_macros" +version = "3.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d846214a9854ef724f3da161b426242d8de7c1fc7de2f89bb1efcb154dca79d" +dependencies = [ + "darling", + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "serialize-to-javascript" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c9823f2d3b6a81d98228151fdeaf848206a7855a7a042bbf9bf870449a66cafb" +dependencies = [ + "serde", + "serde_json", + "serialize-to-javascript-impl", +] + +[[package]] +name = "serialize-to-javascript-impl" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "74064874e9f6a15f04c1f3cb627902d0e6b410abbf36668afa873c61889f1763" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "servo_arc" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d98238b800e0d1576d8b6e3de32827c2d74bee68bb97748dcf5071fb53965432" +dependencies = [ + "nodrop", + "stable_deref_trait", +] + +[[package]] +name = "sha1" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[package]] +name = "sha2" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[package]] +name = "shared_child" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09fa9338aed9a1df411814a5b2252f7cd206c55ae9bf2fa763f8de84603aa60c" +dependencies = [ + "libc", + "windows-sys 0.59.0", +] + +[[package]] +name = "shlex" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" + +[[package]] +name = "signal-hook-registry" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a9e9e0b4211b72e7b8b6e85c807d36c212bdb33ea8587f7569562a84df5465b1" +dependencies = [ + "libc", +] + +[[package]] +name = "simd-adler32" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d66dc143e6b11c1eddc06d5c423cfc97062865baf299914ab64caa38182078fe" + +[[package]] +name = "simd_helpers" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95890f873bec569a0362c235787f3aca6e1e887302ba4840839bcc6459c42da6" +dependencies = [ + "quote", +] + +[[package]] +name = "simdutf8" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3a9fe34e3e7a50316060351f37187a3f546bce95496156754b601a5fa71b76e" + +[[package]] +name = "siphasher" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38b58827f4464d87d377d175e90bf58eb00fd8716ff0a62f80356b5e61555d0d" + +[[package]] +name = "slab" +version = "0.4.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67" +dependencies = [ + "autocfg", +] + +[[package]] +name = "smallvec" +version = "1.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" + +[[package]] +name = "socket2" +version = "0.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce305eb0b4296696835b71df73eb912e0f1ffd2556a501fcede6e0c50349191c" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "softbuffer" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "18051cdd562e792cad055119e0cdb2cfc137e44e3987532e0f9659a77931bb08" +dependencies = [ + "bytemuck", + "cfg_aliases 0.2.1", + "core-graphics 0.24.0", + "foreign-types 0.5.0", + "js-sys", + "log", + "objc2", + "objc2-foundation", + "objc2-quartz-core", + "raw-window-handle", + "redox_syscall", + "wasm-bindgen", + "web-sys", + "windows-sys 0.59.0", +] + +[[package]] +name = "soup3" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "471f924a40f31251afc77450e781cb26d55c0b650842efafc9c6cbd2f7cc4f9f" +dependencies = [ + "futures-channel", + "gio", + "glib", + "libc", + "soup3-sys", +] + +[[package]] +name = "soup3-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ebe8950a680a12f24f15ebe1bf70db7af98ad242d9db43596ad3108aab86c27" +dependencies = [ + "gio-sys", + "glib-sys", + "gobject-sys", + "libc", + "system-deps", +] + +[[package]] +name = "spin" +version = "0.9.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67" +dependencies = [ + "lock_api", +] + +[[package]] +name = "stable_deref_trait" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3" + +[[package]] +name = "starship-battery" +version = "0.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "725bc1c7374f435ef65746eb1a5789cb7d02b8e997f9a3edf979bfb42da68311" +dependencies = [ + "cfg-if", + "core-foundation 0.9.4", + "lazycell", + "libc", + "mach2", + "nix 0.28.0", + "num-traits", + "uom 0.36.0", + "winapi", +] + +[[package]] +name = "static_assertions" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" + +[[package]] +name = "string_cache" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f91138e76242f575eb1d3b38b4f1362f10d3a43f47d182a5b359af488a02293b" +dependencies = [ + "new_debug_unreachable", + "once_cell", + "parking_lot", + "phf_shared 0.10.0", + "precomputed-hash", + "serde", +] + +[[package]] +name = "string_cache_codegen" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6bb30289b722be4ff74a408c3cc27edeaad656e06cb1fe8fa9231fa59c728988" +dependencies = [ + "phf_generator 0.10.0", + "phf_shared 0.10.0", + "proc-macro2", + "quote", +] + +[[package]] +name = "strip-ansi-escapes" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55ff8ef943b384c414f54aefa961dd2bd853add74ec75e7ac74cf91dba62bcfa" +dependencies = [ + "vte", +] + +[[package]] +name = "strsim" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623" + +[[package]] +name = "strsim" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" + +[[package]] +name = "strum" +version = "0.26.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fec0f0aef304996cf250b31b5a10dee7980c85da9d759361292b8bca5a18f06" + +[[package]] +name = "strum_macros" +version = "0.26.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c6bee85a5a24955dc440386795aa378cd9cf82acd5f764469152d2270e581be" +dependencies = [ + "heck 0.5.0", + "proc-macro2", + "quote", + "rustversion", + "syn 2.0.86", +] + +[[package]] +name = "subtle" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" + +[[package]] +name = "swift-rs" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4057c98e2e852d51fdcfca832aac7b571f6b351ad159f9eda5db1655f8d0c4d7" +dependencies = [ + "base64 0.21.7", + "serde", + "serde_json", +] + +[[package]] +name = "syn" +version = "1.0.109" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "syn" +version = "2.0.86" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e89275301d38033efb81a6e60e3497e734dfcc62571f2854bf4b16690398824c" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "syn_derive" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1329189c02ff984e9736652b1631330da25eaa6bc639089ed4915d25446cbe7b" +dependencies = [ + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "sync_wrapper" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" + +[[package]] +name = "sync_wrapper" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394" +dependencies = [ + "futures-core", +] + +[[package]] +name = "sys-locale" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8eab9a99a024a169fe8a903cf9d4a3b3601109bcc13bd9e3c6fff259138626c4" +dependencies = [ + "libc", +] + +[[package]] +name = "sysinfo" +version = "0.30.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0a5b4ddaee55fb2bea2bf0e5000747e5f5c0de765e5a5ff87f4cd106439f4bb3" +dependencies = [ + "cfg-if", + "core-foundation-sys", + "libc", + "ntapi", + "once_cell", + "rayon", + "windows 0.52.0", +] + +[[package]] +name = "system-configuration" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba3a3adc5c275d719af8cb4272ea1c4a6d668a777f37e115f6d11ddbc1c8e0e7" +dependencies = [ + "bitflags 1.3.2", + "core-foundation 0.9.4", + "system-configuration-sys 0.5.0", +] + +[[package]] +name = "system-configuration" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c879d448e9d986b661742763247d3693ed13609438cf3d006f51f5368a5ba6b" +dependencies = [ + "bitflags 2.6.0", + "core-foundation 0.9.4", + "system-configuration-sys 0.6.0", +] + +[[package]] +name = "system-configuration-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75fb188eb626b924683e3b95e3a48e63551fcfb51949de2f06a9d91dbee93c9" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "system-configuration-sys" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e1d1b10ced5ca923a1fcb8d03e96b8d3268065d724548c0211415ff6ac6bac4" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "system-deps" +version = "6.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3e535eb8dded36d55ec13eddacd30dec501792ff23a0b1682c38601b8cf2349" +dependencies = [ + "cfg-expr", + "heck 0.5.0", + "pkg-config", + "toml 0.8.2", + "version-compare", +] + +[[package]] +name = "tao" +version = "0.30.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63f1f6b2017cc33d7f6fc9c6186a2c0f5dfc985899a7b4fe9e64985c17533db3" +dependencies = [ + "bitflags 2.6.0", + "cocoa 0.26.0", + "core-foundation 0.10.0", + "core-graphics 0.24.0", + "crossbeam-channel", + "dispatch", + "dlopen2", + "dpi", + "gdkwayland-sys", + "gdkx11-sys", + "gtk", + "instant", + "jni", + "lazy_static", + "libc", + "log", + "ndk", + "ndk-context", + "ndk-sys", + "objc", + "once_cell", + "parking_lot", + "raw-window-handle", + "scopeguard", + "tao-macros", + "unicode-segmentation", + "url", + "windows 0.58.0", + "windows-core 0.58.0", + "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.86", +] + +[[package]] +name = "tap" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369" + +[[package]] +name = "tar" +version = "0.4.43" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c65998313f8e17d0d553d28f91a0df93e4dbbbf770279c7bc21ca0f09ea1a1f6" +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.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3889b392db6d32a105d3757230ea0220090b8f94c90d3e60b6c5eb91178ab1b" +dependencies = [ + "anyhow", + "bytes", + "dirs 5.0.1", + "dunce", + "embed_plist", + "futures-util", + "getrandom 0.2.15", + "glob", + "gtk", + "heck 0.5.0", + "http 1.1.0", + "image", + "jni", + "libc", + "log", + "mime", + "muda", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "percent-encoding", + "plist", + "raw-window-handle", + "reqwest 0.12.9", + "serde", + "serde_json", + "serde_repr", + "serialize-to-javascript", + "swift-rs", + "tauri-build", + "tauri-macros", + "tauri-runtime", + "tauri-runtime-wry", + "tauri-utils", + "thiserror", + "tokio", + "tray-icon", + "url", + "urlpattern", + "webkit2gtk", + "webview2-com", + "window-vibrancy", + "windows 0.58.0", +] + +[[package]] +name = "tauri-build" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f96827ccfb1aa40d55d0ded79562d18ba18566657a553f992a982d755148376" +dependencies = [ + "anyhow", + "cargo_toml", + "dirs 5.0.1", + "glob", + "heck 0.5.0", + "json-patch 3.0.1", + "schemars", + "semver", + "serde", + "serde_json", + "tauri-utils", + "tauri-winres", + "toml 0.8.2", + "walkdir", +] + +[[package]] +name = "tauri-codegen" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8947f16f47becd9e9cd39b74ee337fd1981574d78819be18e4384d85e5a0b82f" +dependencies = [ + "base64 0.22.1", + "brotli", + "ico", + "json-patch 2.0.0", + "plist", + "png", + "proc-macro2", + "quote", + "semver", + "serde", + "serde_json", + "sha2", + "syn 2.0.86", + "tauri-utils", + "thiserror", + "time", + "url", + "uuid", + "walkdir", +] + +[[package]] +name = "tauri-icns" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "03b7eb4d0d43724ba9ba6a6717420ee68aee377816a3edbb45db8c18862b1431" +dependencies = [ + "byteorder", + "png", +] + +[[package]] +name = "tauri-macros" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8bd1c8d4a66799d3438747c3a79705cd665a95d6f24cb5f315413ff7a981fe2a" +dependencies = [ + "heck 0.5.0", + "proc-macro2", + "quote", + "syn 2.0.86", + "tauri-codegen", + "tauri-utils", +] + +[[package]] +name = "tauri-plugin" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6fa4e6c94cb1d635f65a770c69e23de1bc054b0e4c554fa037a7cc7676333d39" +dependencies = [ + "anyhow", + "glob", + "plist", + "schemars", + "serde", + "serde_json", + "tauri-utils", + "toml 0.8.2", + "walkdir", +] + +[[package]] +name = "tauri-plugin-cli" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2bccd4692b56822a60874542c7655546c8e7aed3c2e2926166e1498e595bd2a2" +dependencies = [ + "clap", + "log", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "thiserror", +] + +[[package]] +name = "tauri-plugin-clipboard" +version = "2.1.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f811a26526cad7291414c526be85b2d1317681c60949bbe8c10902898b7c48d7" +dependencies = [ + "base64 0.22.1", + "clipboard-rs", + "image", + "serde", + "tauri", + "tauri-plugin", + "thiserror", +] + +[[package]] +name = "tauri-plugin-deep-link" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31a9b5725027c6e9e075b06cb2d5c2cd3b5c29daa8012b404e1ff755cc56082f" +dependencies = [ + "dunce", + "log", + "rust-ini", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "tauri-utils", + "thiserror", + "url", + "windows-registry 0.3.0", + "windows-result 0.2.0", +] + +[[package]] +name = "tauri-plugin-dialog" +version = "2.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4307310e1d2c09ab110235834722e7c2b85099b683e1eb7342ab351b0be5ada3" +dependencies = [ + "log", + "raw-window-handle", + "rfd", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "tauri-plugin-fs", + "thiserror", + "url", +] + +[[package]] +name = "tauri-plugin-fs" +version = "2.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96ba7d46e86db8c830d143ef90ab5a453328365b0cc834c24edea4267b16aba0" +dependencies = [ + "anyhow", + "dunce", + "glob", + "percent-encoding", + "schemars", + "serde", + "serde_json", + "serde_repr", + "tauri", + "tauri-plugin", + "thiserror", + "url", + "uuid", +] + +[[package]] +name = "tauri-plugin-global-shortcut" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2c15fb7f5e4c80a73ce97217dcff27e423f496178cbcb87e13b4efe99eebb550" +dependencies = [ + "global-hotkey", + "log", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "thiserror", +] + +[[package]] +name = "tauri-plugin-http" +version = "2.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c752aee1b00ec3c4d4f440095995d9bd2c640b478f2067d1fba388900b82eb96" +dependencies = [ + "data-url", + "http 1.1.0", + "regex", + "reqwest 0.12.9", + "schemars", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "tauri-plugin-fs", + "thiserror", + "tokio", + "url", + "urlpattern", +] + +[[package]] +name = "tauri-plugin-jarvis" +version = "0.0.0" +dependencies = [ + "anyhow", + "applications", + "axum", + "axum-extra", + "axum-server", + "chrono", + "db", + "flate2", + "ico", + "log", + "mac-security-rs", + "mdns-sd", + "plist", + "prost", + "rust_search", + "serde", + "serde_json", + "strum", + "strum_macros", + "tar", + "tauri", + "tauri-icns", + "tauri-plugin", + "tauri-plugin-clipboard", + "tauri-plugin-network", + "tauri-plugin-store", + "tauri-winres", + "thiserror", + "tokio", + "tonic", + "tonic-build", + "tonic-reflection", + "tower", + "tower-http", + "uuid", + "window-vibrancy", + "zip 1.1.4", +] + +[[package]] +name = "tauri-plugin-log" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a49f2c05d15e6375ab7f7e528b3049150ba4dfafdf61f85e5178d0aef18e3f5" +dependencies = [ + "android_logger", + "byte-unit", + "cocoa 0.26.0", + "fern", + "log", + "objc", + "serde", + "serde_json", + "serde_repr", + "swift-rs", + "tauri", + "tauri-plugin", + "thiserror", + "time", +] + +[[package]] +name = "tauri-plugin-network" +version = "2.0.4" +dependencies = [ + "futures", + "gethostname 0.4.3", + "if-addrs 0.10.2", + "ipnetwork", + "lazy_static", + "mdns-sd", + "network-interface", + "reqwest 0.11.27", + "serde", + "tauri", + "tauri-plugin", + "thiserror", +] + +[[package]] +name = "tauri-plugin-notification" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef492a2d19b6376bb4c9e0c4fab3f3bf8a220ea112d24f35027b737ff55de20c" +dependencies = [ + "log", + "notify-rust", + "rand 0.8.5", + "serde", + "serde_json", + "serde_repr", + "tauri", + "tauri-plugin", + "thiserror", + "time", + "url", +] + +[[package]] +name = "tauri-plugin-os" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fbc5f23a86f37687c7f4fecfdc706b279087bc44f7a46702f7307ff1551ee03a" +dependencies = [ + "gethostname 0.5.0", + "log", + "os_info", + "serde", + "serde_json", + "serialize-to-javascript", + "sys-locale", + "tauri", + "tauri-plugin", + "thiserror", +] + +[[package]] +name = "tauri-plugin-process" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ae06a00087c148962a52814a2d7265b1a0505bced5ffb74f8c284a5f96a4d03d" +dependencies = [ + "tauri", + "tauri-plugin", +] + +[[package]] +name = "tauri-plugin-shell" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ad7880c5586b6b2104be451e3d7fc0f3800c84bda69e9ba81c828f87cb34267" +dependencies = [ + "encoding_rs", + "log", + "open", + "os_pipe", + "regex", + "schemars", + "serde", + "serde_json", + "shared_child", + "tauri", + "tauri-plugin", + "thiserror", + "tokio", +] + +[[package]] +name = "tauri-plugin-shellx" +version = "2.0.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2f35c21a6f9bda23881d6fc82006306ffbeb3aee32b621f3c866bf8402007157" +dependencies = [ + "encoding_rs", + "open", + "os_pipe", + "regex", + "schemars", + "serde", + "serde_json", + "shared_child", + "strip-ansi-escapes", + "tauri", + "tauri-plugin", + "thiserror", + "tokio", +] + +[[package]] +name = "tauri-plugin-single-instance" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a25ac834491d089699a2bc9266a662faf373c9f779f05a2235bc6e4d9e61769a" +dependencies = [ + "log", + "serde", + "serde_json", + "tauri", + "tauri-plugin-deep-link", + "thiserror", + "windows-sys 0.59.0", + "zbus", +] + +[[package]] +name = "tauri-plugin-store" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e9a580be53f04bb62422d239aa798e88522877f58a0d4a0e745f030055a51bb4" +dependencies = [ + "dunce", + "log", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "thiserror", + "tokio", +] + +[[package]] +name = "tauri-plugin-system-info" +version = "2.0.8" +dependencies = [ + "serde", + "starship-battery", + "sysinfo", + "tauri", + "tauri-plugin", + "thiserror", + "uom 0.35.0", +] + +[[package]] +name = "tauri-plugin-updater" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1dd3d2fe0f02bf52eebb5a9d23b987fffac6684646ab6fd683d706dafb18da87" +dependencies = [ + "base64 0.22.1", + "dirs 5.0.1", + "flate2", + "futures-util", + "http 1.1.0", + "infer", + "minisign-verify", + "percent-encoding", + "reqwest 0.12.9", + "semver", + "serde", + "serde_json", + "tar", + "tauri", + "tauri-plugin", + "tempfile", + "thiserror", + "time", + "tokio", + "url", + "windows-sys 0.59.0", + "zip 2.2.0", +] + +[[package]] +name = "tauri-plugin-upload" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "116754130d3f95cf73552a0723376186a8f21607ac5ff350980af87a0eccba73" +dependencies = [ + "futures-util", + "log", + "read-progress-stream", + "reqwest 0.12.9", + "serde", + "serde_json", + "tauri", + "tauri-plugin", + "thiserror", + "tokio", + "tokio-util", +] + +[[package]] +name = "tauri-runtime" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1ef7363e7229ac8d04e8a5d405670dbd43dde8fc4bc3bc56105c35452d03784" +dependencies = [ + "dpi", + "gtk", + "http 1.1.0", + "jni", + "raw-window-handle", + "serde", + "serde_json", + "tauri-utils", + "thiserror", + "url", + "windows 0.58.0", +] + +[[package]] +name = "tauri-runtime-wry" +version = "2.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62fa2068e8498ad007b54d5773d03d57c3ff6dd96f8c8ce58beff44d0d5e0d30" +dependencies = [ + "gtk", + "http 1.1.0", + "jni", + "log", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "percent-encoding", + "raw-window-handle", + "softbuffer", + "tao", + "tauri-runtime", + "tauri-utils", + "url", + "webkit2gtk", + "webview2-com", + "windows 0.58.0", + "wry", +] + +[[package]] +name = "tauri-utils" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fc65d6f5c54e56b66258948a6d9e47a82ea41f4b5a7612bfbdd1634c2913ed0" +dependencies = [ + "brotli", + "cargo_metadata", + "ctor", + "dunce", + "glob", + "html5ever", + "infer", + "json-patch 2.0.0", + "kuchikiki", + "log", + "memchr", + "phf 0.11.2", + "proc-macro2", + "quote", + "regex", + "schemars", + "semver", + "serde", + "serde-untagged", + "serde_json", + "serde_with", + "swift-rs", + "thiserror", + "toml 0.8.2", + "url", + "urlpattern", + "uuid", + "walkdir", +] + +[[package]] +name = "tauri-winres" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5993dc129e544393574288923d1ec447c857f3f644187f4fbf7d9a875fbfc4fb" +dependencies = [ + "embed-resource", + "toml 0.7.8", +] + +[[package]] +name = "tauri-winrt-notification" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f89f5fb70d6f62381f5d9b2ba9008196150b40b75f3068eb24faeddf1c686871" +dependencies = [ + "quick-xml 0.31.0", + "windows 0.56.0", + "windows-version", +] + +[[package]] +name = "tempfile" +version = "3.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0f2c9fc62d0beef6951ccffd757e241266a2c833136efbe35af6cd2567dca5b" +dependencies = [ + "cfg-if", + "fastrand", + "once_cell", + "rustix", + "windows-sys 0.59.0", +] + +[[package]] +name = "tendril" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d24a120c5fc464a3458240ee02c299ebcb9d67b5249c8848b09d639dca8d7bb0" +dependencies = [ + "futf", + "mac", + "utf-8", +] + +[[package]] +name = "thin-slice" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8eaa81235c7058867fa8c0e7314f33dcce9c215f535d1913822a2b3f5e289f3c" + +[[package]] +name = "thiserror" +version = "1.0.66" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d171f59dbaa811dbbb1aee1e73db92ec2b122911a48e1390dfe327a821ddede" +dependencies = [ + "thiserror-impl", +] + +[[package]] +name = "thiserror-impl" +version = "1.0.66" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b08be0f17bd307950653ce45db00cd31200d82b624b36e181337d9c7d92765b5" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "tiff" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba1310fcea54c6a9a4fd1aad794ecc02c31682f6bfbecdf460bf19533eed1e3e" +dependencies = [ + "flate2", + "jpeg-decoder", + "weezl", +] + +[[package]] +name = "time" +version = "0.3.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5dfd88e563464686c916c7e46e623e520ddc6d79fa6641390f2e3fa86e83e885" +dependencies = [ + "deranged", + "itoa 1.0.11", + "libc", + "num-conv", + "num_threads", + "powerfmt", + "serde", + "time-core", + "time-macros", +] + +[[package]] +name = "time-core" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3" + +[[package]] +name = "time-macros" +version = "0.2.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f252a68540fde3a3877aeea552b832b40ab9a69e318efd078774a01ddee1ccf" +dependencies = [ + "num-conv", + "time-core", +] + +[[package]] +name = "tiny-keccak" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2c9d3793400a45f954c52e73d068316d76b6f4e36977e3fcebb13a2721e80237" +dependencies = [ + "crunchy", +] + +[[package]] +name = "tinyvec" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "445e881f4f6d382d5f27c034e25eb92edd7c784ceab92a0937db7f2e9471b938" +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.41.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "145f3413504347a2be84393cc8a7d2fb4d863b375909ea59f2158261aa258bbb" +dependencies = [ + "backtrace", + "bytes", + "libc", + "mio", + "pin-project-lite", + "signal-hook-registry", + "socket2", + "tokio-macros", + "tracing", + "windows-sys 0.52.0", +] + +[[package]] +name = "tokio-io-timeout" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "30b74022ada614a1b4834de765f9bb43877f910cc8ce4be40e89042c9223a8bf" +dependencies = [ + "pin-project-lite", + "tokio", +] + +[[package]] +name = "tokio-macros" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "693d596312e88961bc67d7f1f97af8a70227d9f90c31bba5806eec004978d752" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "tokio-native-tls" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbae76ab933c85776efabc971569dd6119c580d8f5d448769dec1764bf796ef2" +dependencies = [ + "native-tls", + "tokio", +] + +[[package]] +name = "tokio-rustls" +version = "0.24.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c28327cf380ac148141087fbfb9de9d7bd4e84ab5d2c28fbc911d753de8a7081" +dependencies = [ + "rustls 0.21.12", + "tokio", +] + +[[package]] +name = "tokio-rustls" +version = "0.26.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c7bc40d0e5a97695bb96e27995cd3a08538541b0a846f65bba7a359f36700d4" +dependencies = [ + "rustls 0.23.16", + "rustls-pki-types", + "tokio", +] + +[[package]] +name = "tokio-stream" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f4e6ce100d0eb49a2734f8c0812bcd324cf357d21810932c5df6b96ef2b86f1" +dependencies = [ + "futures-core", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "tokio-util" +version = "0.7.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61e7c3654c13bcd040d4a03abee2c75b1d14a37b423cf5a813ceae1cc903ec6a" +dependencies = [ + "bytes", + "futures-core", + "futures-sink", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "toml" +version = "0.7.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd79e69d3b627db300ff956027cc6c3798cef26d22526befdfcd12feeb6d2257" +dependencies = [ + "serde", + "serde_spanned", + "toml_datetime", + "toml_edit 0.19.15", +] + +[[package]] +name = "toml" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "185d8ab0dfbb35cf1399a6344d8484209c088f75f8f68230da55d48d95d43e3d" +dependencies = [ + "serde", + "serde_spanned", + "toml_datetime", + "toml_edit 0.20.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_edit" +version = "0.19.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b5bb770da30e5cbfde35a2d7b9b8a2c4b8ef89548a7a6aeab5c9a576e3e7421" +dependencies = [ + "indexmap 2.6.0", + "serde", + "serde_spanned", + "toml_datetime", + "winnow", +] + +[[package]] +name = "toml_edit" +version = "0.20.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "396e4d48bbb2b7554c944bde63101b5ae446cff6ec4a24227428f15eb72ef338" +dependencies = [ + "indexmap 2.6.0", + "serde", + "serde_spanned", + "toml_datetime", + "winnow", +] + +[[package]] +name = "tonic" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76c4eb7a4e9ef9d4763600161f12f5070b92a578e1b634db88a6887844c91a13" +dependencies = [ + "async-stream", + "async-trait", + "axum", + "base64 0.21.7", + "bytes", + "h2 0.3.26", + "http 0.2.12", + "http-body 0.4.6", + "hyper 0.14.31", + "hyper-timeout", + "percent-encoding", + "pin-project", + "prost", + "tokio", + "tokio-stream", + "tower", + "tower-layer", + "tower-service", + "tracing", +] + +[[package]] +name = "tonic-build" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be4ef6dd70a610078cb4e338a0f79d06bc759ff1b22d2120c2ff02ae264ba9c2" +dependencies = [ + "prettyplease", + "proc-macro2", + "prost-build", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "tonic-reflection" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "548c227bd5c0fae5925812c4ec6c66ffcfced23ea370cb823f4d18f0fc1cb6a7" +dependencies = [ + "prost", + "prost-types", + "tokio", + "tokio-stream", + "tonic", +] + +[[package]] +name = "tower" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8fa9be0de6cf49e536ce1851f987bd21a43b771b09473c3549a6c853db37c1c" +dependencies = [ + "futures-core", + "futures-util", + "indexmap 1.9.3", + "pin-project", + "pin-project-lite", + "rand 0.8.5", + "slab", + "tokio", + "tokio-util", + "tower-layer", + "tower-service", + "tracing", +] + +[[package]] +name = "tower-http" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61c5bb1d698276a2443e5ecfabc1008bf15a36c12e6a7176e7bf089ea9131140" +dependencies = [ + "bitflags 2.6.0", + "bytes", + "futures-core", + "futures-util", + "http 0.2.12", + "http-body 0.4.6", + "http-range-header", + "httpdate", + "mime", + "mime_guess", + "percent-encoding", + "pin-project-lite", + "tokio", + "tokio-util", + "tower-layer", + "tower-service", + "tracing", +] + +[[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.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" +dependencies = [ + "log", + "pin-project-lite", + "tracing-attributes", + "tracing-core", +] + +[[package]] +name = "tracing-attributes" +version = "0.1.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "tracing-core" +version = "0.1.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" +dependencies = [ + "once_cell", +] + +[[package]] +name = "tray-icon" +version = "0.19.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c92af36a182b46206723bdf8a7942e20838cde1cf062e5b97854d57eb01763b" +dependencies = [ + "core-graphics 0.24.0", + "crossbeam-channel", + "dirs 5.0.1", + "libappindicator", + "muda", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "once_cell", + "png", + "serde", + "thiserror", + "windows-sys 0.59.0", +] + +[[package]] +name = "trim-in-place" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "343e926fc669bc8cde4fa3129ab681c63671bae288b1f1081ceee6d9d37904fc" + +[[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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e13db2e0ccd5e14a544e8a246ba2312cd25223f616442d7f2cb0e3db614236e" + +[[package]] +name = "typenum" +version = "1.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" + +[[package]] +name = "uds_windows" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "89daebc3e6fd160ac4aa9fc8b3bf71e1f74fbf92367ae71fb83a037e8bf164b9" +dependencies = [ + "memoffset", + "tempfile", + "winapi", +] + +[[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.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e51b68083f157f853b6379db119d1c1be0e6e4dec98101079dec41f6f5cf6df" + +[[package]] +name = "unicode-bidi" +version = "0.3.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ab17db44d7388991a428b2ee655ce0c212e862eff1768a455c58f9aad6e7893" + +[[package]] +name = "unicode-ident" +version = "1.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e91b56cd4cadaeb79bbf1a5645f6b4f8dc5bde8834ad5894a8db35fda9efa1fe" + +[[package]] +name = "unicode-normalization" +version = "0.1.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5033c97c4262335cded6d6fc3e5c18ab755e1a3dc96376350f3d8e9f009ad956" +dependencies = [ + "tinyvec", +] + +[[package]] +name = "unicode-segmentation" +version = "1.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6ccf251212114b54433ec949fd6a7841275f9ada20dddd2f29e9ceea4501493" + +[[package]] +name = "untrusted" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" + +[[package]] +name = "uom" +version = "0.35.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8362194c7a9845a7a7f3562173d6e1da3f24f7132018cb78fe77a5b4474187b2" +dependencies = [ + "num-traits", + "serde", + "typenum", +] + +[[package]] +name = "uom" +version = "0.36.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ffd36e5350a65d112584053ee91843955826bf9e56ec0d1351214e01f6d7cd9c" +dependencies = [ + "num-traits", + "typenum", +] + +[[package]] +name = "url" +version = "2.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22784dbdf76fdde8af1aeda5622b546b422b6fc585325248a2bf9f5e41e94d6c" +dependencies = [ + "form_urlencoded", + "idna 0.5.0", + "percent-encoding", + "serde", +] + +[[package]] +name = "urlencoding" +version = "2.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "daf8dba3b7eb870caf1ddeed7bc9d2a049f3cfdfae7cb521b087cc33ae4c49da" + +[[package]] +name = "urlpattern" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70acd30e3aa1450bc2eece896ce2ad0d178e9c079493819301573dae3c37ba6d" +dependencies = [ + "regex", + "serde", + "unic-ucd-ident", + "url", +] + +[[package]] +name = "utf-8" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" + +[[package]] +name = "utf8-width" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "86bd8d4e895da8537e5315b8254664e6b769c4ff3db18321b297a1e7004392e3" + +[[package]] +name = "utf8parse" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" + +[[package]] +name = "uuid" +version = "1.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8c5f0a0af699448548ad1a2fbf920fb4bee257eae39953ba95cb84891a0446a" +dependencies = [ + "getrandom 0.2.15", + "serde", +] + +[[package]] +name = "v_frame" +version = "0.3.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d6f32aaa24bacd11e488aa9ba66369c7cd514885742c9fe08cfe85884db3e92b" +dependencies = [ + "aligned-vec", + "num-traits", + "wasm-bindgen", +] + +[[package]] +name = "value-bag" +version = "1.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3ef4c4aa54d5d05a279399bfa921ec387b7aba77caf7a682ae8d86785b8fdad2" + +[[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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "852e951cb7832cb45cb1169900d19760cfa39b82bc0ea9c0e5a14ae88411c98b" + +[[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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3b17ae1f6c8a2b28506cd96d412eebf83b4a0ff2cbefeeb952f2f9dfa44ba18" +dependencies = [ + "cc", + "libc", +] + +[[package]] +name = "vte" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f5022b5fbf9407086c180e9557be968742d839e68346af7792b8592489732197" +dependencies = [ + "utf8parse", + "vte_generate_state_changes", +] + +[[package]] +name = "vte_generate_state_changes" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e369bee1b05d510a7b4ed645f5faa90619e05437111783ea5848f28d97d3c2e" +dependencies = [ + "proc-macro2", + "quote", +] + +[[package]] +name = "walkdir" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b" +dependencies = [ + "same-file", + "winapi-util", +] + +[[package]] +name = "want" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e" +dependencies = [ + "try-lock", +] + +[[package]] +name = "wasi" +version = "0.9.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519" + +[[package]] +name = "wasi" +version = "0.11.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" + +[[package]] +name = "wasm-bindgen" +version = "0.2.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "128d1e363af62632b8eb57219c8fd7877144af57558fb2ef0368d0087bddeb2e" +dependencies = [ + "cfg-if", + "once_cell", + "wasm-bindgen-macro", +] + +[[package]] +name = "wasm-bindgen-backend" +version = "0.2.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb6dd4d3ca0ddffd1dd1c9c04f94b868c37ff5fac97c30b97cff2d74fce3a358" +dependencies = [ + "bumpalo", + "log", + "once_cell", + "proc-macro2", + "quote", + "syn 2.0.86", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc7ec4f8827a71586374db3e87abdb5a2bb3a15afed140221307c3ec06b1f63b" +dependencies = [ + "cfg-if", + "js-sys", + "wasm-bindgen", + "web-sys", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e79384be7f8f5a9dd5d7167216f022090cf1f9ec128e6e6a482a2cb5c5422c56" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26c6ab57572f7a24a4985830b120de1594465e5d500f24afe89e16b4e833ef68" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", + "wasm-bindgen-backend", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "65fc09f10666a9f147042251e0dda9c18f166ff7de300607007e96bdebc1068d" + +[[package]] +name = "wasm-streams" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15053d8d85c7eccdbefef60f06769760a563c7f0a9d6902a13d35c7800b0ad65" +dependencies = [ + "futures-util", + "js-sys", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "wayland-backend" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "056535ced7a150d45159d3a8dc30f91a2e2d588ca0b23f70e56033622b8016f6" +dependencies = [ + "cc", + "downcast-rs", + "rustix", + "scoped-tls", + "smallvec", + "wayland-sys", +] + +[[package]] +name = "wayland-client" +version = "0.31.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b66249d3fc69f76fd74c82cc319300faa554e9d865dab1f7cd66cc20db10b280" +dependencies = [ + "bitflags 2.6.0", + "rustix", + "wayland-backend", + "wayland-scanner", +] + +[[package]] +name = "wayland-protocols" +version = "0.32.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7cd0ade57c4e6e9a8952741325c30bf82f4246885dca8bf561898b86d0c1f58e" +dependencies = [ + "bitflags 2.6.0", + "wayland-backend", + "wayland-client", + "wayland-scanner", +] + +[[package]] +name = "wayland-scanner" +version = "0.31.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "597f2001b2e5fc1121e3d5b9791d3e78f05ba6bfa4641053846248e3a13661c3" +dependencies = [ + "proc-macro2", + "quick-xml 0.36.2", + "quote", +] + +[[package]] +name = "wayland-sys" +version = "0.31.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "efa8ac0d8e8ed3e3b5c9fc92c7881406a268e11555abe36493efabe649a29e09" +dependencies = [ + "dlib", + "log", + "pkg-config", +] + +[[package]] +name = "web-sys" +version = "0.3.72" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6488b90108c040df0fe62fa815cbdee25124641df01814dd7282749234c6112" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "webkit2gtk" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76b1bc1e54c581da1e9f179d0b38512ba358fb1af2d634a1affe42e37172361a" +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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62daa38afc514d1f8f12b8693d30d5993ff77ced33ce30cd04deebc267a6d57c" +dependencies = [ + "bitflags 1.3.2", + "cairo-sys-rs", + "gdk-sys", + "gio-sys", + "glib-sys", + "gobject-sys", + "gtk-sys", + "javascriptcore-rs-sys", + "libc", + "pkg-config", + "soup3-sys", + "system-deps", +] + +[[package]] +name = "webpki-roots" +version = "0.26.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "841c67bff177718f1d4dfefde8d8f0e78f9b6589319ba88312f567fc5841a958" +dependencies = [ + "rustls-pki-types", +] + +[[package]] +name = "webview2-com" +version = "0.33.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6f61ff3d9d0ee4efcb461b14eb3acfda2702d10dc329f339303fc3e57215ae2c" +dependencies = [ + "webview2-com-macros", + "webview2-com-sys", + "windows 0.58.0", + "windows-core 0.58.0", + "windows-implement 0.58.0", + "windows-interface 0.58.0", +] + +[[package]] +name = "webview2-com-macros" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d228f15bba3b9d56dde8bddbee66fa24545bd17b48d5128ccf4a8742b18e431" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "webview2-com-sys" +version = "0.33.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3a3e2eeb58f82361c93f9777014668eb3d07e7d174ee4c819575a9208011886" +dependencies = [ + "thiserror", + "windows 0.58.0", + "windows-core 0.58.0", +] + +[[package]] +name = "weezl" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53a85b86a771b1c87058196170769dd264f66c0782acf1ae6cc51bfd64b39082" + +[[package]] +name = "widestring" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c168940144dd21fd8046987c16a46a33d5fc84eec29ef9dcddc2ac9e31526b7c" + +[[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.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" +dependencies = [ + "windows-sys 0.59.0", +] + +[[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.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3ea403deff7b51fff19e261330f71608ff2cdef5721d72b64180bb95be7c4150" +dependencies = [ + "objc2", + "objc2-app-kit", + "objc2-foundation", + "raw-window-handle", + "windows-sys 0.59.0", + "windows-version", +] + +[[package]] +name = "windows" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e48a53791691ab099e5e2ad123536d0fff50652600abaf43bbf952894110d0be" +dependencies = [ + "windows-core 0.52.0", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows" +version = "0.56.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1de69df01bdf1ead2f4ac895dc77c9351aefff65b2f3db429a343f9cbf05e132" +dependencies = [ + "windows-core 0.56.0", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows" +version = "0.58.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd04d41d93c4992d421894c18c8b43496aa748dd4c081bac0dc93eb0489272b6" +dependencies = [ + "windows-core 0.58.0", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-core" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-core" +version = "0.56.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4698e52ed2d08f8658ab0c39512a7c00ee5fe2688c65f8c0a4f06750d729f2a6" +dependencies = [ + "windows-implement 0.56.0", + "windows-interface 0.56.0", + "windows-result 0.1.2", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-core" +version = "0.58.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ba6d44ec8c2591c134257ce647b7ea6b20335bf6379a27dac5f1641fcf59f99" +dependencies = [ + "windows-implement 0.58.0", + "windows-interface 0.58.0", + "windows-result 0.2.0", + "windows-strings 0.1.0", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-implement" +version = "0.56.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6fc35f58ecd95a9b71c4f2329b911016e6bec66b3f2e6a4aad86bd2e99e2f9b" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "windows-implement" +version = "0.58.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2bbd5b46c938e506ecbce286b6628a02171d56153ba733b6c741fc627ec9579b" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "windows-interface" +version = "0.56.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08990546bf4edef8f431fa6326e032865f27138718c587dc21bc0265bbcb57cc" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "windows-interface" +version = "0.58.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "053c4c462dc91d3b1504c6fe5a726dd15e216ba718e84a0e46a88fbe5ded3515" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "windows-registry" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e400001bb720a623c1c69032f8e3e4cf09984deec740f007dd2b03ec864804b0" +dependencies = [ + "windows-result 0.2.0", + "windows-strings 0.1.0", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-registry" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bafa604f2104cf5ae2cc2db1dee84b7e6a5d11b05f737b60def0ffdc398cbc0a" +dependencies = [ + "windows-result 0.2.0", + "windows-strings 0.2.0", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-result" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e383302e8ec8515204254685643de10811af0ed97ea37210dc26fb0032647f8" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-result" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d1043d8214f791817bab27572aaa8af63732e11bf84aa21a45a78d6c317ae0e" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-strings" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4cd9b125c486025df0eabcb585e62173c6c9eddcec5d117d3b6e8c30e2ee4d10" +dependencies = [ + "windows-result 0.2.0", + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-strings" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "978d65aedf914c664c510d9de43c8fd85ca745eaff1ed53edf409b479e441663" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-sys" +version = "0.45.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75283be5efb2831d37ea142365f009c02ec203cd29a3ebecbc093d52315b66d0" +dependencies = [ + "windows-targets 0.42.2", +] + +[[package]] +name = "windows-sys" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" +dependencies = [ + "windows-targets 0.48.5", +] + +[[package]] +name = "windows-sys" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-sys" +version = "0.59.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-targets" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e5180c00cd44c9b1c88adb3693291f1cd93605ded80c250a75d472756b4d071" +dependencies = [ + "windows_aarch64_gnullvm 0.42.2", + "windows_aarch64_msvc 0.42.2", + "windows_i686_gnu 0.42.2", + "windows_i686_msvc 0.42.2", + "windows_x86_64_gnu 0.42.2", + "windows_x86_64_gnullvm 0.42.2", + "windows_x86_64_msvc 0.42.2", +] + +[[package]] +name = "windows-targets" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c" +dependencies = [ + "windows_aarch64_gnullvm 0.48.5", + "windows_aarch64_msvc 0.48.5", + "windows_i686_gnu 0.48.5", + "windows_i686_msvc 0.48.5", + "windows_x86_64_gnu 0.48.5", + "windows_x86_64_gnullvm 0.48.5", + "windows_x86_64_msvc 0.48.5", +] + +[[package]] +name = "windows-targets" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973" +dependencies = [ + "windows_aarch64_gnullvm 0.52.6", + "windows_aarch64_msvc 0.52.6", + "windows_i686_gnu 0.52.6", + "windows_i686_gnullvm", + "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-version" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6998aa457c9ba8ff2fb9f13e9d2a930dabcea28f1d0ab94d687d8b3654844515" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-win" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "58e23e33622b3b52f948049acbec9bcc34bf6e26d74176b88941f213c75cf2dc" +dependencies = [ + "error-code", +] + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "597a5118570b68bc08d8d59125332c54f1ba9d9adeedeef5b99b02ba2b0698f8" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e08e8864a60f06ef0d0ff4ba04124db8b0fb3be5776a5cd47641e942e58c4d43" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469" + +[[package]] +name = "windows_i686_gnu" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c61d927d8da41da96a81f029489353e68739737d3beca43145c8afec9a31a84f" + +[[package]] +name = "windows_i686_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" + +[[package]] +name = "windows_i686_gnu" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b" + +[[package]] +name = "windows_i686_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66" + +[[package]] +name = "windows_i686_msvc" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44d840b6ec649f480a41c8d80f9c65108b92d89345dd94027bfe06ac444d1060" + +[[package]] +name = "windows_i686_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" + +[[package]] +name = "windows_i686_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8de912b8b8feb55c064867cf047dda097f92d51efad5b491dfb98f6bbb70cb36" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26d41b46a36d453748aedef1486d5c7a85db22e56aff34643984ea85514e94a3" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9aec5da331524158c6d1a4ac0ab1541149c0b9505fde06423b02f5ef0106b9f0" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.52.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" + +[[package]] +name = "winnow" +version = "0.5.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f593a95398737aeed53e489c785df13f3618e41dbcd6718c6addbf1395aa6876" +dependencies = [ + "memchr", +] + +[[package]] +name = "winreg" +version = "0.50.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "524e57b2c537c0f9b1e69f1965311ec12182b4122e45035b1508cd24d2adadb1" +dependencies = [ + "cfg-if", + "windows-sys 0.48.0", +] + +[[package]] +name = "winreg" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a277a57398d4bfa075df44f501a17cfdf8542d224f0d36095a2adc7aee4ef0a5" +dependencies = [ + "cfg-if", + "windows-sys 0.48.0", +] + +[[package]] +name = "wry" +version = "0.46.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cd5cdf57c66813d97601181349c63b96994b3074fc3d7a31a8cce96e968e3bbd" +dependencies = [ + "base64 0.22.1", + "block2", + "crossbeam-channel", + "dpi", + "dunce", + "gdkx11", + "gtk", + "html5ever", + "http 1.1.0", + "javascriptcore-rs", + "jni", + "kuchikiki", + "libc", + "ndk", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "objc2-ui-kit", + "objc2-web-kit", + "once_cell", + "percent-encoding", + "raw-window-handle", + "sha2", + "soup3", + "tao-macros", + "thiserror", + "webkit2gtk", + "webkit2gtk-sys", + "webview2-com", + "windows 0.58.0", + "windows-core 0.58.0", + "windows-version", + "x11-dl", +] + +[[package]] +name = "wyz" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05f360fc0b24296329c78fda852a1e9ae82de9cf7b27dae4b7f62f118f77b9ed" +dependencies = [ + "tap", +] + +[[package]] +name = "x11" +version = "2.21.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "502da5464ccd04011667b11c435cb992822c2c0dbde1770c988480d312a0db2e" +dependencies = [ + "libc", + "pkg-config", +] + +[[package]] +name = "x11-dl" +version = "2.21.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38735924fedd5314a6e548792904ed8c6de6636285cb9fec04d5b1db85c1516f" +dependencies = [ + "libc", + "once_cell", + "pkg-config", +] + +[[package]] +name = "x11rb" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d91ffca73ee7f68ce055750bf9f6eca0780b8c85eff9bc046a3b0da41755e12" +dependencies = [ + "gethostname 0.4.3", + "rustix", + "x11rb-protocol", +] + +[[package]] +name = "x11rb-protocol" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec107c4503ea0b4a98ef47356329af139c0a4f7750e621cf2973cd3385ebcb3d" + +[[package]] +name = "xattr" +version = "1.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8da84f1a25939b27f6820d92aed108f83ff920fdf11a7b19366c27c4cda81d4f" +dependencies = [ + "libc", + "linux-raw-sys", + "rustix", +] + +[[package]] +name = "xdg-home" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec1cdab258fb55c0da61328dc52c8764709b249011b2cad0454c72f0bf10a1f6" +dependencies = [ + "libc", + "windows-sys 0.59.0", +] + +[[package]] +name = "zbus" +version = "4.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b8e3d6ae3342792a6cc2340e4394334c7402f3d793b390d2c5494a4032b3030" +dependencies = [ + "async-broadcast", + "async-executor", + "async-fs", + "async-io", + "async-lock", + "async-process", + "async-recursion", + "async-task", + "async-trait", + "blocking", + "derivative", + "enumflags2", + "event-listener", + "futures-core", + "futures-sink", + "futures-util", + "hex", + "nix 0.27.1", + "ordered-stream", + "rand 0.8.5", + "serde", + "serde_repr", + "sha1", + "static_assertions", + "tokio", + "tracing", + "uds_windows", + "windows-sys 0.52.0", + "xdg-home", + "zbus_macros", + "zbus_names", + "zvariant", +] + +[[package]] +name = "zbus_macros" +version = "4.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7a3e850ff1e7217a3b7a07eba90d37fe9bb9e89a310f718afcde5885ca9b6d7" +dependencies = [ + "proc-macro-crate 1.3.1", + "proc-macro2", + "quote", + "regex", + "syn 1.0.109", + "zvariant_utils", +] + +[[package]] +name = "zbus_names" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4b9b1fef7d021261cc16cba64c351d291b715febe0fa10dc3a443ac5a5022e6c" +dependencies = [ + "serde", + "static_assertions", + "zvariant", +] + +[[package]] +name = "zerocopy" +version = "0.7.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0" +dependencies = [ + "byteorder", + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.7.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "zeroize" +version = "1.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" +dependencies = [ + "zeroize_derive", +] + +[[package]] +name = "zeroize_derive" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.86", +] + +[[package]] +name = "zip" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9cc23c04387f4da0374be4533ad1208cbb091d5c11d070dfef13676ad6497164" +dependencies = [ + "aes", + "arbitrary", + "bzip2", + "constant_time_eq", + "crc32fast", + "crossbeam-utils", + "deflate64", + "displaydoc", + "flate2", + "hmac", + "indexmap 2.6.0", + "lzma-rs", + "num_enum", + "pbkdf2", + "sha1", + "thiserror", + "time", + "zopfli", + "zstd", +] + +[[package]] +name = "zip" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc5e4288ea4057ae23afc69a4472434a87a2495cafce6632fd1c4ec9f5cf3494" +dependencies = [ + "aes", + "arbitrary", + "bzip2", + "constant_time_eq", + "crc32fast", + "crossbeam-utils", + "deflate64", + "displaydoc", + "flate2", + "hmac", + "indexmap 2.6.0", + "lzma-rs", + "memchr", + "pbkdf2", + "rand 0.8.5", + "sha1", + "thiserror", + "time", + "zeroize", + "zopfli", + "zstd", +] + +[[package]] +name = "zopfli" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5019f391bac5cf252e93bbcc53d039ffd62c7bfb7c150414d61369afe57e946" +dependencies = [ + "bumpalo", + "crc32fast", + "lockfree-object-pool", + "log", + "once_cell", + "simd-adler32", +] + +[[package]] +name = "zstd" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fcf2b778a664581e31e389454a7072dab1647606d44f7feea22cd5abb9c9f3f9" +dependencies = [ + "zstd-safe", +] + +[[package]] +name = "zstd-safe" +version = "7.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "54a3ab4db68cea366acc5c897c7b4d4d1b8994a9cd6e6f841f8964566a419059" +dependencies = [ + "zstd-sys", +] + +[[package]] +name = "zstd-sys" +version = "2.0.13+zstd.1.5.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38ff0f21cfee8f97d94cef41359e0c89aa6113028ab0291aa8ca0038995a95aa" +dependencies = [ + "cc", + "pkg-config", +] + +[[package]] +name = "zune-core" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f423a2c17029964870cfaabb1f13dfab7d092a62a29a89264f4d36990ca414a" + +[[package]] +name = "zune-inflate" +version = "0.2.54" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73ab332fe2f6680068f3582b16a24f90ad7096d5d39b974d1c0aff0125116f02" +dependencies = [ + "simd-adler32", +] + +[[package]] +name = "zune-jpeg" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "16099418600b4d8f028622f73ff6e3deaabdff330fb9a2a131dea781ee8b0768" +dependencies = [ + "zune-core", +] + +[[package]] +name = "zvariant" +version = "4.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4e09e8be97d44eeab994d752f341e67b3b0d80512a8b315a0671d47232ef1b65" +dependencies = [ + "endi", + "enumflags2", + "serde", + "static_assertions", + "url", + "zvariant_derive", +] + +[[package]] +name = "zvariant_derive" +version = "4.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72a5857e2856435331636a9fbb415b09243df4521a267c5bedcd5289b4d5799e" +dependencies = [ + "proc-macro-crate 1.3.1", + "proc-macro2", + "quote", + "syn 1.0.109", + "zvariant_utils", +] + +[[package]] +name = "zvariant_utils" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "00bedb16a193cc12451873fee2a1bc6550225acece0e36f333e68326c73c8172" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] diff --git a/Cargo.toml b/Cargo.toml new file mode 100644 index 0000000..0dbdc11 --- /dev/null +++ b/Cargo.toml @@ -0,0 +1,28 @@ +[workspace] +resolver = "2" + +members = [ + "apps/desktop/src-tauri", + "packages/tauri-plugins/jarvis", + "packages/db", + "packages/mac-security-rs", + "packages/tauri-plugins/jarvis", +] + +[workspace.dependencies] +serde = { version = "1", features = ["derive"] } +anyhow = "1.0.86" +serde_json = "1" +tokio = { version = "1.0", features = ["macros", "rt-multi-thread", "signal"] } +mdns-sd = "0.11.1" +tauri-plugin-network = { path = "./vendors/tauri-plugin-network" } +tauri-plugin-clipboard = "2.1.8" +mac-security-rs = { path = "./packages/mac-security-rs" } +log = "0.4.22" +strum = "0.26" +strum_macros = "0.26" +chrono = "0.4.38" +applications = { path = "./vendors/applications-rs" } +tauri-plugin-jarvis = { path = "./packages/tauri-plugins/jarvis" } +tauri-plugin-system-info = { path = "./vendors/tauri-plugin-system-info" } +db = { path = "./packages/db" } diff --git a/README.md b/README.md index 9c6a408..7996bd9 100644 --- a/README.md +++ b/README.md @@ -5,4 +5,3 @@ - Website: https://kunkun.sh/ - Documentation: https://docs.kunkun.sh/ - diff --git a/apps/desktop/README.md b/apps/desktop/README.md index 37c369f..9bffde2 100644 --- a/apps/desktop/README.md +++ b/apps/desktop/README.md @@ -5,4 +5,3 @@ - Keep all components as modular as possible - Don't use global store directly in components, pass them through context or props instead - The components may be exported as a package and used by other projects such as docs, extension - diff --git a/apps/desktop/package.json b/apps/desktop/package.json index 8fc5c26..8384d98 100644 --- a/apps/desktop/package.json +++ b/apps/desktop/package.json @@ -1,42 +1,52 @@ { - "name": "@kksh/desktop", - "version": "0.1.0", - "description": "", - "type": "module", - "scripts": { - "dev": "vite dev", - "build": "vite build", - "preview": "vite preview", - "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json", - "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch", - "tauri": "tauri" - }, - "license": "MIT", - "dependencies": { - "@tauri-apps/api": "^2", - "@tauri-apps/plugin-shell": "^2" - }, - "devDependencies": { - "@kksh/ui": "workspace:*", - "@sveltejs/adapter-static": "^3.0.5", - "@sveltejs/kit": "^2.7.0", - "@sveltejs/vite-plugin-svelte": "^4.0.0", - "@tailwindcss/aspect-ratio": "^0.4.2", - "@tailwindcss/container-queries": "^0.1.1", - "@tailwindcss/forms": "^0.5.9", - "@tailwindcss/typography": "^0.5.15", - "@tauri-apps/cli": "^2", - "autoprefixer": "^10.4.20", - "clsx": "^2.1.1", - "embla-carousel-svelte": "^8.3.1", - "formsnap": "^1.0.1", - "tailwind-merge": "^2.5.4", - "tailwind-variants": "^0.2.1", - "tailwindcss": "^3.4.9", - "tailwindcss-animate": "^1.0.7", - "tslib": "^2.8.0", - "typescript": "^5.5.0", - "vaul-svelte": "^0.3.2", - "vite": "^5.4.10" - } + "name": "@kksh/desktop", + "version": "0.1.9-beta.8", + "description": "", + "type": "module", + "scripts": { + "dev": "vite dev", + "build": "vite build", + "preview": "vite preview", + "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json", + "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch", + "tauri": "tauri" + }, + "license": "MIT", + "dependencies": { + "@kksh/extension": "workspace:*", + "@kksh/supabase": "workspace:*", + "@kksh/ui": "workspace:*", + "@kksh/utils": "workspace:*", + "@std/semver": "npm:@jsr/std__semver@^1.0.3", + "@tauri-apps/api": "^2", + "@tauri-apps/plugin-shell": "^2", + "bits-ui": "1.0.0-next.36", + "lucide-svelte": "^0.454.0", + "svelte-radix": "^2.0.1", + "svelte-sonner": "^0.3.28", + "sveltekit-superforms": "^2.20.0" + }, + "devDependencies": { + "@kksh/types": "workspace:*", + "@sveltejs/adapter-static": "^3.0.6", + "@sveltejs/kit": "^2.7.4", + "@sveltejs/vite-plugin-svelte": "^4.0.0", + "@tailwindcss/aspect-ratio": "^0.4.2", + "@tailwindcss/container-queries": "^0.1.1", + "@tailwindcss/forms": "^0.5.9", + "@tailwindcss/typography": "^0.5.15", + "@tauri-apps/cli": "^2.0.4", + "autoprefixer": "^10.4.20", + "clsx": "^2.1.1", + "embla-carousel-svelte": "^8.3.1", + "formsnap": "^1.0.1", + "tailwind-merge": "^2.5.4", + "tailwind-variants": "^0.2.1", + "tailwindcss": "^3.4.14", + "tailwindcss-animate": "^1.0.7", + "tslib": "^2.8.1", + "typescript": "^5.6.3", + "vaul-svelte": "^0.3.2", + "vite": "^5.4.10" + } } diff --git a/apps/desktop/src-tauri/Cargo.toml b/apps/desktop/src-tauri/Cargo.toml index 7a86e7d..9293907 100644 --- a/apps/desktop/src-tauri/Cargo.toml +++ b/apps/desktop/src-tauri/Cargo.toml @@ -1,10 +1,9 @@ [package] name = "kunkun" -version = "0.1.0" -description = "A Tauri App" -authors = ["you"] +version = "0.0.0" +description = "Kunkun Desktop App" +authors = ["Huakun"] edition = "2021" - # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [lib] @@ -15,11 +14,50 @@ name = "kunkun_lib" crate-type = ["staticlib", "cdylib", "rlib"] [build-dependencies] -tauri-build = { version = "2", features = [] } +tauri-build = { version = "2.0.2", features = [] } [dependencies] -tauri = { version = "2", features = [] } +tauri = { version = "2.0.6", features = [ "macos-private-api", + "image-png", + "image-ico", + "tray-icon", + "devtools", +] } tauri-plugin-shell = "2" -serde = { version = "1", features = ["derive"] } -serde_json = "1" +serde = { workspace = true } +serde_json = { workspace = true } +anyhow = { workspace = true } +tokio = { workspace = true } +mdns-sd = { workspace = true } +chrono = { workspace = true } +log = { workspace = true } +urlencoding = "2.1.3" +tauri-plugin-process = "2.0.1" +tauri-plugin-shellx = "2.0.11" +tauri-plugin-fs = "2.0.1" +tauri-plugin-dialog = "2.0.1" +tauri-plugin-notification = "2.0.1" +tauri-plugin-os = "2.0.1" +tauri-plugin-http = "2.0.1" +tauri-plugin-upload = "2.0.1" +tauri-plugin-jarvis = { workspace = true } +tauri-plugin-network = { workspace = true } +tauri-plugin-system-info = { workspace = true } +tauri-plugin-clipboard = { workspace = true } +tauri-plugin-store = "2.1.0" +tauri-plugin-deep-link = "2" +tauri-plugin-log = { version = "2.0.1", features = ["colored"] } +zip = "2.1.3" +# tauri-plugin-devtools = "2.0.0" +[target."cfg(target_os = \"macos\")".dependencies] +cocoa = "0.24.1" +mac-security-rs = { workspace = true } +objc = "0.2.7" + + +[target."cfg(not(any(target_os = \"android\", target_os = \"ios\")))".dependencies] +tauri-plugin-cli = "2" +tauri-plugin-global-shortcut = "2.0.1" +tauri-plugin-single-instance = { version = "2", features = ["deep-link"] } +tauri-plugin-updater = "2.0.2" diff --git a/apps/desktop/src-tauri/capabilities/default.json b/apps/desktop/src-tauri/capabilities/default.json index 3bb4cc4..176a04d 100644 --- a/apps/desktop/src-tauri/capabilities/default.json +++ b/apps/desktop/src-tauri/capabilities/default.json @@ -1,10 +1,164 @@ { - "$schema": "../gen/schemas/desktop-schema.json", - "identifier": "default", - "description": "Capability for the main window", - "windows": ["main"], - "permissions": [ - "core:default", - "shell:allow-open" - ] + "$schema": "../gen/schemas/desktop-schema.json", + "identifier": "default", + "description": "Capability for the main window", + "windows": ["main*"], + "permissions": [ + { + "identifier": "http:default", + "allow": [ + { + "url": "https://*" + }, + { + "url": "http://*" + }, + { + "url": "http://*:*" + } + ] + }, + "os:default", + "os:allow-platform", + "core:path:default", + "core:event:default", + "core:window:default", + "core:window:allow-start-dragging", + "core:window:allow-set-focus", + "core:window:allow-toggle-maximize", + "core:window:allow-internal-toggle-maximize", + "core:window:allow-close", + "core:window:allow-create", + "core:window:allow-set-decorations", + "core:window:allow-show", + "core:window:allow-hide", + "core:window:allow-destroy", + "core:image:default", + "core:webview:default", + "core:webview:allow-create-webview", + "core:webview:allow-create-webview-window", + "core:app:default", + "core:resources:default", + "core:menu:default", + "core:tray:default", + "core:tray:allow-new", + "core:tray:allow-set-tooltip", + "core:tray:allow-set-icon", + "core:tray:allow-set-menu", + "core:tray:allow-set-title", + "core:tray:allow-set-visible", + "core:tray:allow-set-show-menu-on-left-click", + "notification:default", + "clipboard:monitor-all", + "clipboard:read-all", + "clipboard:write-all", + "fs:default", + "fs:allow-app-meta", + "fs:allow-home-read-recursive", + "shellx:allow-execute", + "shellx:allow-open", + "shellx:allow-kill", + "shellx:allow-spawn", + "shellx:allow-stdin-write", + "shellx:allow-fix-path-env", + "dialog:default", + "dialog:allow-open", + "dialog:allow-confirm", + "dialog:allow-save", + "dialog:allow-message", + "dialog:allow-ask", + "global-shortcut:default", + "global-shortcut:allow-is-registered", + "global-shortcut:allow-register", + "global-shortcut:allow-unregister", + "jarvis:allow-all", + "store:default", + "store:allow-clear", + "store:allow-delete", + "store:allow-entries", + "store:allow-get", + "store:allow-has", + "store:allow-keys", + "store:allow-length", + "store:allow-load", + "store:allow-reset", + "store:allow-save", + "store:allow-set", + "store:allow-values", + "log:default", + "updater:default", + "log:allow-log", + "fs:allow-exists", + "fs:allow-stat", + "fs:read-all", + "fs:write-all", + "fs:allow-rename", + "fs:scope-temp-recursive", + "fs:scope-temp", + "fs:scope-home-recursive", + "fs:allow-mkdir", + "fs:allow-app-write-recursive", + "fs:allow-app-read-recursive", + "fs:allow-appcache-write", + "fs:allow-appcache-write-recursive", + "fs:allow-appconfig-write", + "fs:allow-home-write-recursive", + "fs:allow-appdata-read-recursive", + "fs:allow-appdata-write-recursive", + { + "identifier": "fs:scope", + "allow": [ + { + "path": "$DESKTOP" + }, + { + "path": "$DESKTOP/**" + }, + { + "path": "$DOWNLOAD" + }, + { + "path": "$DOWNLOAD/**" + }, + { + "path": "$DOCUMENT" + }, + { + "path": "$DOCUMENT/**" + }, + { + "path": "$TEMP/**" + }, + { + "path": "$TEMP" + } + ] + }, + "notification:allow-is-permission-granted", + "notification:allow-notify", + "notification:allow-request-permission", + "global-shortcut:allow-register-all", + "global-shortcut:allow-unregister-all", + "clipboard:allow-clear", + "cli:default", + "upload:default", + "process:default", + "system-info:allow-all", + "shell:default", + { + "identifier": "shell:allow-spawn", + "allow": [ + { + "name": "deno", + "cmd": "deno", + "args": [ + { + "validator": ".+" + } + ] + } + ] + }, + "deep-link:default" + ] } diff --git a/apps/desktop/src-tauri/src/lib.rs b/apps/desktop/src-tauri/src/lib.rs index f91b35e..05ed166 100644 --- a/apps/desktop/src-tauri/src/lib.rs +++ b/apps/desktop/src-tauri/src/lib.rs @@ -1,14 +1,241 @@ -// Learn more about Tauri commands at https://tauri.app/develop/calling-rust/ -#[tauri::command] -fn greet(name: &str) -> String { - format!("Hello, {}! You've been greeted from Rust!", name) -} +use std::{path::PathBuf, sync::Mutex}; +mod setup; +pub mod utils; +use log; +#[cfg(target_os = "macos")] +use tauri::ActivationPolicy; +use tauri::Manager; +use tauri_plugin_deep_link::DeepLinkExt; +use tauri_plugin_jarvis::{ + db::JarvisDB, + server::Protocol, + utils::{ + path::{get_default_extensions_dir, get_kunkun_db_path}, + settings::AppSettings, + }, +}; +pub use tauri_plugin_log::fern::colors::ColoredLevelConfig; +use tauri_plugin_store::{StoreBuilder, StoreExt}; +use utils::server::tauri_file_server; #[cfg_attr(mobile, tauri::mobile_entry_point)] pub fn run() { - tauri::Builder::default() + let context = tauri::generate_context!(); + let mut builder = tauri::Builder::default(); + + #[cfg(debug_assertions)] + { + println!("Install crabnebula devtools"); + // let devtools = tauri_plugin_devtools::init(); // initialize the plugin as early as possible + // builder = builder.plugin(devtools); + } + // #[cfg(not(debug_assertions))] + // { + // builder = builder.plugin( + // tauri_plugin_log::Builder::new() + // .targets(utils::log::get_log_targets()) + // .level(utils::log::get_log_level()) + // .filter(|metadata| !metadata.target().starts_with("mdns_sd")) + // .with_colors(ColoredLevelConfig::default()) + // .max_file_size(10_000_000) // max 10MB + // .format(|out, message, record| { + // out.finish(format_args!( + // "{}[{}] {}", + // chrono::Local::now().format("[%Y-%m-%d][%H:%M:%S]"), + // // record.target(), + // record.level(), + // message + // )) + // }) + // .build(), + // ); + // } + + let shell_unlocked = true; + builder = builder + .plugin(tauri_plugin_single_instance::init(|app, args, cwd| { + let _ = app + .get_webview_window("main") + .expect("no main window") + .set_focus(); + })) + .plugin( + tauri_plugin_log::Builder::new() + .targets(utils::log::get_log_targets()) + .level(utils::log::get_log_level()) + .filter(|metadata| !metadata.target().starts_with("mdns_sd")) + .with_colors(ColoredLevelConfig::default()) + .max_file_size(10_000_000) // max 10MB + .format(|out, message, record| { + out.finish(format_args!( + "{}[{}] {}", + chrono::Local::now().format("[%Y-%m-%d][%H:%M:%S]"), + // record.target(), + record.level(), + message + )) + }) + .build(), + ) + .plugin(tauri_plugin_cli::init()) + .plugin(tauri_plugin_deep_link::init()) .plugin(tauri_plugin_shell::init()) - .invoke_handler(tauri::generate_handler![greet]) - .run(tauri::generate_context!()) + .plugin(tauri_plugin_os::init()) + .plugin(tauri_plugin_process::init()) + .plugin(tauri_plugin_upload::init()) + .plugin(tauri_plugin_updater::Builder::new().build()) + .plugin(tauri_plugin_store::Builder::default().build()) + .plugin(tauri_plugin_global_shortcut::Builder::new().build()) + .plugin(tauri_plugin_dialog::init()) + .plugin(tauri_plugin_http::init()) + .plugin(tauri_plugin_notification::init()) + .plugin(tauri_plugin_fs::init()) + .plugin(tauri_plugin_shellx::init(shell_unlocked)) + .plugin(tauri_plugin_jarvis::init()) + .plugin(tauri_plugin_clipboard::init()) + .plugin(tauri_plugin_network::init()) + .plugin(tauri_plugin_system_info::init()); + + let app = builder + .register_uri_scheme_protocol("appicon", |_app, request| { + let url = &request.uri().path()[1..]; + let url = urlencoding::decode(url).unwrap().to_string(); + let path = PathBuf::from(url); + return tauri_plugin_jarvis::utils::icns::load_icon(path); + }) + .register_uri_scheme_protocol("ext", |app, request| { + let app_handle = app.app_handle(); + // app_handle. + let win_label = app.webview_label(); + let jarvis_state = app_handle.state::(); + let window_ext_map = jarvis_state.window_label_ext_map.lock().unwrap(); + match window_ext_map.get(win_label) { + Some(ext) => { + // let app_state = app_handle.state::(); + // let extension_path = app_state.extension_path.lock().unwrap().clone(); + // tauri_file_server(app_handle, request, extension_path) + tauri_file_server(app_handle, request, ext.path.clone(), ext.dist.clone()) + } + None => tauri::http::Response::builder() + .status(tauri::http::StatusCode::NOT_FOUND) + .header("Access-Control-Allow-Origin", "*") + .body("Extension Not Found".as_bytes().to_vec()) + .unwrap(), + } + }) + .setup(|app| { + setup::window::setup_window(app.handle()); + setup::tray::create_tray(app.handle())?; + #[cfg(all(not(target_os = "macos"), debug_assertions))] + { + app.deep_link().register("kunkun")?; + } + // setup::deeplink::setup_deeplink(app); + // #[cfg(all(target_os = "macos", debug_assertions))] + // app.set_activation_policy(ActivationPolicy::Accessory); + // let mut store = StoreBuilder::new("appConfig.bin").build(app.handle().clone()); + // let store = app.handle().store_builder("appConfig.json").build()?; + + // let app_settings = match AppSettings::load_from_store(&store) { + // Ok(settings) => settings, + // Err(_) => AppSettings::default(), + // }; + // let dev_extension_path: Option = app_settings.dev_extension_path.clone(); + let my_port = tauri_plugin_network::network::scan::find_available_port_from_list( + tauri_plugin_jarvis::server::CANDIDATE_PORTS.to_vec(), + ) + .unwrap(); + log::info!("Jarvis Server Port: {}", my_port); + // log::info!( + // "App Settings Dev Extension Path: {:?}", + // app_settings.dev_extension_path.clone(), + // ); + app.manage(tauri_plugin_jarvis::server::http::Server::new( + app.handle().clone(), + my_port, + Protocol::Http, + // Protocol::Https, + )); + app.manage(tauri_plugin_jarvis::model::app_state::AppState {}); + tauri_plugin_jarvis::setup::server::setup_server(app.handle())?; // start the server + + let mdns = tauri_plugin_jarvis::setup::peer_discovery::setup_mdns(my_port)?; + tauri_plugin_jarvis::setup::peer_discovery::handle_mdns_service_evt( + app.handle(), + mdns.browse()?, + ); + + /* ----------------------------- Database Setup ----------------------------- */ + // setup::db::setup_db(app)?; + /* ------------------------- Clipboard History Setup ------------------------ */ + let db_path = get_kunkun_db_path(app.app_handle())?; + let db_key: Option = None; + let jarvis_db = JarvisDB::new(db_path.clone(), db_key.clone())?; + // The clipboard extension should be created in setup_db, ext is guaranteed to be Some + let ext = jarvis_db.get_unique_extension_by_identifier( + tauri_plugin_jarvis::constants::KUNKUN_CLIPBOARD_EXT_IDENTIFIER, + )?; + + app.manage( + tauri_plugin_jarvis::model::clipboard_history::ClipboardHistory::new( + jarvis_db, + ext.unwrap().ext_id, + ), + ); + let (clipboard_update_tx, clipboard_update_rx) = tokio::sync::broadcast::channel::< + tauri_plugin_jarvis::model::clipboard_history::Record, + >(10); + /* --------------------------- Cliipboard Listener -------------------------- */ + setup::clipboard::setup_clipboard_listener( + &app.app_handle(), + clipboard_update_tx.clone(), + ); + app.state::() + .start_monitor(app.app_handle().clone())?; + setup::clipboard::setup_clipboard_update_handler(app.app_handle(), clipboard_update_rx); + + #[cfg(debug_assertions)] // only include this code on debug builds + { + let window = app.get_webview_window("main").unwrap(); + window.open_devtools(); + // window.close_devtools(); + } + + let main_window = app.get_webview_window("main").unwrap(); + std::thread::spawn(move || { + // this is a backup plan, if frontend is not properly loaded, show() will not be called, then we need to call it manually from rust after a long delay + std::thread::sleep(std::time::Duration::from_secs(1)); + main_window.show().unwrap(); + }); + Ok(()) + }) + .build(context) .expect("error while running tauri application"); + app.run(|_app_handle, event| match event { + // tauri::RunEvent::Exit => todo!(), + // tauri::RunEvent::ExitRequested { code, api, .. } => todo!(), + // tauri::RunEvent::WindowEvent { label, event, .. } => todo!(), + // tauri::RunEvent::WebviewEvent { label, event, .. } => todo!(), + // tauri::RunEvent::Ready => todo!(), + // tauri::RunEvent::Resumed => todo!(), + // tauri::RunEvent::MainEventsCleared => todo!(), + // tauri::RunEvent::Opened { urls } => todo!(), + // tauri::RunEvent::MenuEvent(_) => todo!(), + // tauri::RunEvent::TrayIconEvent(_) => todo!(), + #[cfg(target_os = "macos")] + tauri::RunEvent::Reopen { + has_visible_windows, + .. + } => { + _app_handle + .webview_windows() + .iter() + .for_each(|(label, window)| { + window.show().unwrap(); + }); + // let main_window = _app_handle.get_webview_window("main").unwrap(); + // main_window.show().unwrap(); + } + _ => {} + }); } diff --git a/apps/desktop/src-tauri/src/setup/clipboard.rs b/apps/desktop/src-tauri/src/setup/clipboard.rs new file mode 100644 index 0000000..bd3bb2e --- /dev/null +++ b/apps/desktop/src-tauri/src/setup/clipboard.rs @@ -0,0 +1,83 @@ +use log::error; +use std::time::SystemTime; +use tauri::{AppHandle, Emitter, Listener, Manager, Runtime}; +use tauri_plugin_jarvis::model::clipboard_history; +use tokio::sync::broadcast::{Receiver, Sender}; + +pub fn setup_clipboard_listener( + app: &tauri::AppHandle, + clipboard_update_tx: Sender, +) { + let app2 = app.clone(); + app.listen( + "plugin:clipboard://clipboard-monitor/update", + move |_event| { + let clipboard = app2.state::(); + // let clipboard_history = app2.state::(); + let available_types = clipboard.available_types().unwrap(); + let mut rec = clipboard_history::Record { + timestamp: SystemTime::now() + .duration_since(SystemTime::UNIX_EPOCH) + .unwrap() + .as_millis(), + content_type: clipboard_history::ClipboardContentType::Text, + value: "test".to_string(), + text: "test".to_string(), + }; + if available_types.image { + rec.content_type = clipboard_history::ClipboardContentType::Image; + rec.value = clipboard.read_image_base64().unwrap(); + rec.text = "Image".to_string(); + } else if available_types.html { + rec.content_type = clipboard_history::ClipboardContentType::Html; + let html = clipboard.read_html().unwrap(); + rec.value = html.clone(); + if available_types.text { + rec.text = clipboard.read_text().unwrap(); + } else { + rec.text = html; + } + } else if available_types.rtf { + rec.content_type = clipboard_history::ClipboardContentType::Rtf; + let rtf = clipboard.read_rtf().unwrap(); + rec.value = rtf.clone(); + if available_types.text { + rec.text = clipboard.read_text().unwrap(); + } else { + rec.text = rtf; + } + } else if available_types.files { + rec.content_type = clipboard_history::ClipboardContentType::Text; + rec.value = clipboard.read_files().unwrap().join("\n"); + rec.text = "Files".to_string(); + } else if available_types.text { + rec.content_type = clipboard_history::ClipboardContentType::Text; + rec.value = clipboard.read_text().unwrap(); + rec.text = rec.value.clone(); + } + match clipboard_update_tx.send(rec) { + Ok(_) => {} + Err(e) => { + error!("Error sending clipboard record: {:?}", e); + } + }; + // clipboard_history.add_record(rec); + // app2.emit("new_clipboard_item_added", ()).unwrap(); + }, + ); +} + +pub fn setup_clipboard_update_handler( + app_handle: &AppHandle, + mut clipboard_update_rx: Receiver, +) { + let app_handle = app_handle.clone(); + tauri::async_runtime::spawn(async move { + loop { + let record = clipboard_update_rx.recv().await.unwrap(); + let clipboard_history = app_handle.state::(); + clipboard_history.add_record(record).unwrap(); + app_handle.emit("new_clipboard_item_added", ()).unwrap(); + } + }); +} diff --git a/apps/desktop/src-tauri/src/setup/deeplink.rs b/apps/desktop/src-tauri/src/setup/deeplink.rs new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/apps/desktop/src-tauri/src/setup/deeplink.rs @@ -0,0 +1 @@ + diff --git a/apps/desktop/src-tauri/src/setup/mod.rs b/apps/desktop/src-tauri/src/setup/mod.rs new file mode 100644 index 0000000..309dc48 --- /dev/null +++ b/apps/desktop/src-tauri/src/setup/mod.rs @@ -0,0 +1,4 @@ +pub mod clipboard; +pub mod deeplink; +pub mod tray; +pub mod window; diff --git a/apps/desktop/src-tauri/src/setup/tray.rs b/apps/desktop/src-tauri/src/setup/tray.rs new file mode 100644 index 0000000..dfb6a04 --- /dev/null +++ b/apps/desktop/src-tauri/src/setup/tray.rs @@ -0,0 +1,72 @@ +use tauri::{ + menu::{Menu, MenuItem}, + tray::{MouseButton, MouseButtonState, TrayIconBuilder, TrayIconEvent}, + Manager, Runtime, +}; + +fn toggle_main_window(app: &tauri::AppHandle) { + let main_win = app.get_webview_window("main"); + + if let Some(main_win) = main_win { + if main_win.is_visible().unwrap() { + main_win.hide().unwrap(); + } else { + main_win.show().unwrap(); + main_win.set_focus().unwrap(); + } + } +} + +pub fn create_tray(app: &tauri::AppHandle) -> tauri::Result<()> { + let toggle_i = MenuItem::with_id(app, "toggle", "Toggle", true, None::<&str>)?; + let quit_i = MenuItem::with_id(app, "quit", "Quit", true, None::<&str>)?; + let menu1 = Menu::with_items(app, &[&toggle_i, &quit_i])?; + let _ = TrayIconBuilder::with_id("tray-1") + .tooltip("Kunkun") + .icon(app.default_window_icon().unwrap().clone()) + .menu(&menu1) + .menu_on_left_click(true) + .on_tray_icon_event(move |icon, event: TrayIconEvent| { + // println!("on tray icon event: {:?}", event); + match event { + TrayIconEvent::Click { + button_state, + button, + .. + } => match button { + MouseButton::Left => match button_state { + MouseButtonState::Up => { + toggle_main_window(icon.app_handle()); + } + _ => {} + }, + _ => {} + }, + + _ => {} + } + }) + .menu_on_left_click(false) + .on_menu_event(|app, event| match event.id.as_ref() { + "toggle" => { + toggle_main_window(app); + } + "quit" => { + app.exit(0); + } + _ => { + println!("unknown menu item: {:?}", event.id); + } + }) + // .on_menu_event(move |app, event| { + // println!("event: {:?}", event); + // let main_win = app.get_webview_window("main"); + // if let Some(main_win) = main_win { + // main_win.show().unwrap(); + // main_win.set_focus().unwrap(); + // } + // }) + .build(app); + + Ok(()) +} diff --git a/apps/desktop/src-tauri/src/setup/window.rs b/apps/desktop/src-tauri/src/setup/window.rs new file mode 100644 index 0000000..470a69f --- /dev/null +++ b/apps/desktop/src-tauri/src/setup/window.rs @@ -0,0 +1,59 @@ +use tauri::{is_dev, App, AppHandle, LogicalSize, Manager, Runtime, Size, WebviewWindow, Window}; + +#[cfg(target_os = "macos")] +use cocoa::appkit::{NSWindow, NSWindowButton, NSWindowStyleMask, NSWindowTitleVisibility}; + +#[cfg(target_os = "macos")] +use objc::runtime::YES; + +pub trait WindowExt { + #[cfg(target_os = "macos")] + fn set_transparent_titlebar(&self, title_transparent: bool, remove_toolbar: bool); +} + +impl WindowExt for WebviewWindow { + #[cfg(target_os = "macos")] + fn set_transparent_titlebar(&self, title_transparent: bool, remove_tool_bar: bool) { + use objc::{msg_send, sel, sel_impl}; + + unsafe { + let id = self.ns_window().unwrap() as cocoa::base::id; + NSWindow::setTitlebarAppearsTransparent_(id, cocoa::base::YES); + let mut style_mask = id.styleMask(); + style_mask.set( + NSWindowStyleMask::NSFullSizeContentViewWindowMask, + title_transparent, + ); + + id.setStyleMask_(style_mask); + + if remove_tool_bar { + let close_button = id.standardWindowButton_(NSWindowButton::NSWindowCloseButton); + let _: () = msg_send![close_button, setHidden: YES]; + let min_button = + id.standardWindowButton_(NSWindowButton::NSWindowMiniaturizeButton); + let _: () = msg_send![min_button, setHidden: YES]; + let zoom_button = id.standardWindowButton_(NSWindowButton::NSWindowZoomButton); + let _: () = msg_send![zoom_button, setHidden: YES]; + } + + id.setTitleVisibility_(if title_transparent { + NSWindowTitleVisibility::NSWindowTitleHidden + } else { + NSWindowTitleVisibility::NSWindowTitleVisible + }); + + id.setTitlebarAppearsTransparent_(if title_transparent { + cocoa::base::YES + } else { + cocoa::base::NO + }); + } + } +} + +pub fn setup_window(app: &AppHandle) { + let window = app.get_webview_window("main").unwrap(); + #[cfg(target_os = "macos")] + window.set_transparent_titlebar(true, true); +} diff --git a/apps/desktop/src-tauri/src/utils/log.rs b/apps/desktop/src-tauri/src/utils/log.rs new file mode 100644 index 0000000..e88330b --- /dev/null +++ b/apps/desktop/src-tauri/src/utils/log.rs @@ -0,0 +1,54 @@ +use log::info; +use tauri::{AppHandle, Manager}; +use tauri_plugin_log::{Target as LogTarget, TargetKind}; + +pub fn get_log_targets() -> Vec { + #[cfg(debug_assertions)] + let log_targets = vec![ + LogTarget::new(TargetKind::Stdout), + LogTarget::new(TargetKind::LogDir { + file_name: Some(format!( + "dev:kunkun-{}", + chrono::Local::now().format("%Y-%m-%d") + )), + }), + LogTarget::new(TargetKind::Webview), + ]; + #[cfg(not(debug_assertions))] + let log_targets = vec![ + LogTarget::new(TargetKind::Stdout), + LogTarget::new(TargetKind::LogDir { + file_name: Some(format!( + "kunkun-{}", + chrono::Local::now().format("%Y-%m-%d") + )), + }), + ]; + log_targets +} + +pub fn get_log_level() -> log::LevelFilter { + #[cfg(debug_assertions)] + return log::LevelFilter::Debug; + #[cfg(not(debug_assertions))] + return log::LevelFilter::Info; +} + +/// Remove log files that are older than 3 days +pub fn clear_old_log_files(app_handle: &AppHandle) -> Result<(), tauri::Error> { + let log_dir = app_handle.path().app_log_dir()?; + let files = std::fs::read_dir(log_dir)?; + for file in files { + let file = file?; + let path = file.path(); + let metadata = std::fs::metadata(&path)?; + let modified_datetime: chrono::DateTime = metadata.modified()?.into(); + let now = chrono::Local::now(); + let duration = now.signed_duration_since(modified_datetime); + if duration.num_days() > 3 { + info!("Removing old log file: {:?}", path); + std::fs::remove_file(&path)?; + } + } + Ok(()) +} diff --git a/apps/desktop/src-tauri/src/utils/mod.rs b/apps/desktop/src-tauri/src/utils/mod.rs new file mode 100644 index 0000000..bcd190d --- /dev/null +++ b/apps/desktop/src-tauri/src/utils/mod.rs @@ -0,0 +1,2 @@ +pub mod log; +pub mod server; diff --git a/apps/desktop/src-tauri/src/utils/server.rs b/apps/desktop/src-tauri/src/utils/server.rs new file mode 100644 index 0000000..53b161c --- /dev/null +++ b/apps/desktop/src-tauri/src/utils/server.rs @@ -0,0 +1,125 @@ +use std::path::PathBuf; + +use tauri::{AppHandle, Manager}; + +pub fn tauri_file_server( + app: &AppHandle, + request: tauri::http::Request>, + extension_folder_path: PathBuf, + dist: Option, +) -> tauri::http::Response> { + // let host = request.uri().host().unwrap(); + // let host_parts: Vec<&str> = host.split(".").collect(); + // if host_parts.len() != 3 { + // return tauri::http::Response::builder() + // .status(tauri::http::StatusCode::NOT_FOUND) + // .header("Access-Control-Allow-Origin", "*") + // .body("Invalid Host".as_bytes().to_vec()) + // .unwrap(); + // } + // expect 3 parts, ext_identifier, dist and ext_type + // let ext_identifier = host_parts[0]; + // let dist = host_parts[1]; + // let ext_type = host_parts[2]; // ext or dev-ext + // let app_state = app.state::(); + // let app_state: tauri:State = app.state(); + // let extension_folder_path: Option = match ext_type { + // "ext" => Some(app_state.extension_path.lock().unwrap().clone()), + // "dev-ext" => app_state.dev_extension_path.lock().unwrap().clone(), + // _ => None, + // }; + // let extension_folder_path = match extension_folder_path { + // Some(path) => path, + // None => { + // return tauri::http::Response::builder() + // .status(tauri::http::StatusCode::NOT_FOUND) + // .header("Access-Control-Allow-Origin", "*") + // .body("Extension Folder Not Found".as_bytes().to_vec()) + // .unwrap() + // } + // }; + println!("dist: {:?}", dist); + let path = &request.uri().path()[1..]; // skip the first / + let path = urlencoding::decode(path).unwrap().to_string(); + let mut url_file_path = extension_folder_path; + // .join(ext_identifier) + match dist { + Some(dist) => url_file_path = url_file_path.join(dist), + None => {} + } + url_file_path = url_file_path.join(path); + println!("url_file_path: {:?}", url_file_path); + // check if it's file or directory, if file and exist, return file, if directory, return index.html, if neither, check .html + if url_file_path.is_file() { + // println!("1st case url_file_path: {:?}", url_file_path); + let mime_type = match url_file_path.extension().and_then(std::ffi::OsStr::to_str) { + Some("js") => "application/javascript", + Some("html") => "text/html", + Some("css") => "text/css", + _ => "application/octet-stream", + }; + return tauri::http::Response::builder() + .status(tauri::http::StatusCode::OK) + .header("Access-Control-Allow-Origin", "*") + .header("Content-Type", mime_type) + .body(std::fs::read(url_file_path).unwrap()) + .unwrap(); + } else if url_file_path.is_dir() { + /* + * there are two cases: + * 1. directory conntains a index.html, then return index.html + * 2. directory has a sibling file with .html extension, return that file + */ + let index_html_path = url_file_path.join("index.html"); + if index_html_path.is_file() { + // println!("2nd case index_html_path: {:?}", index_html_path); + return tauri::http::Response::builder() + .status(tauri::http::StatusCode::OK) + .header("Access-Control-Allow-Origin", "*") + .body(std::fs::read(index_html_path).unwrap()) + .unwrap(); + } + // check if path has a sibling file with .html extension + // get folder name + match url_file_path.file_name() { + Some(folder_name) => { + let parent_path = url_file_path.parent().unwrap(); + let html_file_path = + parent_path.join(format!("{}.html", folder_name.to_str().unwrap())); + if html_file_path.is_file() { + // println!("3rd case html_file_path: {:?}", html_file_path); + return tauri::http::Response::builder() + .status(tauri::http::StatusCode::OK) + .header("Access-Control-Allow-Origin", "*") + .body(std::fs::read(html_file_path).unwrap()) + .unwrap(); + } + } + None => {} + } + + // check if url_file_path's parent has a file with name folder_name.html + } else { + // file not found, check if end with .html works. if path ends with /, remove the / and check if adding .html makes a file + let mut path_str = url_file_path.to_str().unwrap().to_string(); + if path_str.ends_with("/") { + path_str.pop(); + } + path_str.push_str(".html"); + let path_str = PathBuf::from(path_str); + if path_str.is_file() { + // println!("4rd case path_str: {:?}", path_str); + return tauri::http::Response::builder() + .status(tauri::http::StatusCode::OK) + .header("Access-Control-Allow-Origin", "*") + .body(std::fs::read(path_str).unwrap()) + .unwrap(); + } + } + // println!("5th case file not found"); + return tauri::http::Response::builder() + .status(tauri::http::StatusCode::NOT_FOUND) + .header("Access-Control-Allow-Origin", "*") + .body("file not found".as_bytes().to_vec()) + .unwrap(); +} diff --git a/apps/desktop/src-tauri/tauri.conf.json b/apps/desktop/src-tauri/tauri.conf.json index be86e27..c5579cc 100644 --- a/apps/desktop/src-tauri/tauri.conf.json +++ b/apps/desktop/src-tauri/tauri.conf.json @@ -1,35 +1,66 @@ { - "$schema": "https://schema.tauri.app/config/2", - "productName": "kunkun", - "version": "0.1.0", - "identifier": "sh.kunkun.desktop", - "build": { - "beforeDevCommand": "pnpm dev", - "devUrl": "http://localhost:1420", - "beforeBuildCommand": "pnpm build", - "frontendDist": "../build" - }, - "app": { - "windows": [ - { - "title": "kunkun", - "width": 800, - "height": 600 - } - ], - "security": { - "csp": null - } - }, - "bundle": { - "active": true, - "targets": "all", - "icon": [ - "icons/32x32.png", - "icons/128x128.png", - "icons/128x128@2x.png", - "icons/icon.icns", - "icons/icon.ico" - ] - } + "$schema": "https://schema.tauri.app/config/2", + "productName": "kunkun", + "version": "../package.json", + "identifier": "sh.kunkun.desktop", + "build": { + "beforeDevCommand": "pnpm dev", + "devUrl": "http://localhost:1420", + "beforeBuildCommand": "pnpm build", + "frontendDist": "../build" + }, + "app": { + "macOSPrivateApi": true, + "windows": [ + { + "hiddenTitle": true, + "width": 800, + "height": 600 + } + ], + "security": { + "csp": null + } + }, + "bundle": { + "createUpdaterArtifacts": true, + "fileAssociations": [ + { + "ext": ["kunkun"], + "mimeType": "text/plain", + "description": "Used to install Kunkun Extensions with a installer file", + "role": "Viewer" + } + ], + "active": true, + "targets": "all", + "icon": [ + "icons/32x32.png", + "icons/128x128.png", + "icons/128x128@2x.png", + "icons/icon.icns", + "icons/icon.ico" + ] + }, + "plugins": { + "updater": { + "endpoints": ["https://updater.kunkun.sh"], + "pubkey": "dW50cnVzdGVkIGNvbW1lbnQ6IG1pbmlzaWduIHB1YmxpYyBrZXk6IDc1NENCRjZFM0JBOEQ0ODMKUldTRDFLZzdicjlNZFhHS0ZKYk13WkdZUTFUM01LNjkvVW5Bb2x1SnB1R0crbFRuMnlRSlJ0STgK" + }, + "deep-link": { + "desktop": { + "schemes": ["kunkun"] + } + }, + "cli": { + "description": "Kunkun CLI", + "args": [ + { + "short": "v", + "name": "verbose", + "description": "Verbosity level" + } + ] + } + } } diff --git a/apps/desktop/src/lib/cmds/builtin.ts b/apps/desktop/src/lib/cmds/builtin.ts new file mode 100644 index 0000000..a3bf77b --- /dev/null +++ b/apps/desktop/src/lib/cmds/builtin.ts @@ -0,0 +1,206 @@ +import { appState } from "@/stores" +import type { BuiltinCmd } from "@kksh/ui/types" +import { dev } from "$app/environment" +import { goto } from "$app/navigation" + +export const builtinCmds: BuiltinCmd[] = [ + { + name: "Store", + iconifyIcon: "streamline:store-2-solid", + description: "Go to Extension Store", + function: async () => { + appState.clearSearchTerm() + goto("/extension/store") + } + }, + // { + // name: "Sign In", + // iconifyIcon: "mdi:login-variant", + // description: "", + // function: async () => { + // goto("/auth") + // } + // }, + // { + // name: "Sign Out", + // iconifyIcon: "mdi:logout-variant", + // description: "", + // function: async () => { + // const supabase = useSupabaseClient() + // supabase.auth.signOut() + // } + // }, + // { + // name: "Show Draggable Area", + // iconifyIcon: "mingcute:move-fill", + // description: "", + // function: async () => { + // // select all html elements with attribute data-tauri-drag-region + // const elements = document.querySelectorAll("[data-tauri-drag-region]") + // elements.forEach((el) => { + // el.classList.add("bg-red-500/30") + // }) + // setTimeout(() => { + // elements.forEach((el) => { + // el.classList.remove("bg-red-500/30") + // }) + // }, 2_000) + // } + // }, + // { + // name: "Add Dev Extension", + // iconifyIcon: "lineicons:dev", + // description: "", + // function: async () => { + // const appStateStore = useAppStateStore() + // appStateStore.setSearchTermSync("") + // goto("/add-dev-ext") + // } + // }, + // { + // name: "Kunkun Version", + // iconifyIcon: "stash:version-solid", + // description: "", + // function: async () => { + // toast.success(`Kunkun Version: ${await getVersion()}`) + // } + // }, + { + name: "Set Dev Extension Path", + iconifyIcon: "lineicons:dev", + description: "", + function: async () => { + // const appStateStore = useAppStateStore() + appState.clearSearchTerm() + goto("/settings/set-dev-ext-path") + } + } + // { + // name: "Extension Window Troubleshooter", + // iconifyIcon: "material-symbols:window-outline", + // description: "", + // function: async () => { + // const appStateStore = useAppStateStore() + // appStateStore.setSearchTermSync("") + // // goto("/window-troubleshooter") + // const winLabel = `main:window-troubleshooter-${uuidv4()}` + // console.log(winLabel) + // new WebviewWindow(winLabel, { + // url: "/window-troubleshooter", + // title: "Window Troubleshooter" + // }) + // } + // }, + // { + // name: "Extension Permission Inspector", + // iconifyIcon: "hugeicons:inspect-code", + // description: "", + // function: async () => { + // const appStateStore = useAppStateStore() + // appStateStore.setSearchTermSync("") + // goto("/ext-permission-inspector") + // } + // }, + // { + // name: "Extension Loading Troubleshooter", + // iconifyIcon: "material-symbols:troubleshoot", + // description: "", + // function: async () => { + // const appStateStore = useAppStateStore() + // appStateStore.setSearchTermSync("") + // goto("/extension-load-troubleshooter") + // } + // }, + // { + // name: "Create Quicklink", + // iconifyIcon: "material-symbols:link", + // description: "Create a Quicklink", + // function: async () => { + // const appStateStore = useAppStateStore() + // appStateStore.setSearchTermSync("") + // goto("/create-quicklink") + // } + // }, + // { + // name: "Settings", + // iconifyIcon: "solar:settings-linear", + // description: "Open Settings", + // function: async () => { + // const windows = await getAllWebviewWindows() + // const found = windows.find((w) => w.label === SettingsWindowLabel) + // if (found) { + // ElNotification.error("Settings Page is already open") + // } else { + // const win = await newSettingsPage() + // setTimeout(() => { + // // this is a backup, if window is not properly loaded, + // // the show() will not be called within setting page, we call it here with a larger delay, + // // at least the window will be shown + // win.show() + // }, 800) + // } + // const appStateStore = useAppStateStore() + // appStateStore.setSearchTermSync("") + // } + // }, + // { + // name: "Check Update", + // iconifyIcon: "material-symbols:update", + // description: "Check for updates", + // function: async () => { + // checkUpdateAndInstall() + // } + // }, + // { + // name: "Check Beta Update", + // iconifyIcon: "material-symbols:update", + // description: "Check for Beta updates", + // function: async () => { + // checkUpdateAndInstall(true) + // } + // }, + // { + // name: "Reload", + // iconifyIcon: "tabler:reload", + // description: "Reload this page", + // function: async () => { + // location.reload() + // } + // }, + // { + // name: "Dance", + // iconifyIcon: "mdi:dance-pole", + // description: "Dance", + // function: async () => { + // goto("/dance") + // } + // }, + // { + // name: "Quit Kunkun", + // iconifyIcon: "emojione:cross-mark-button", + // description: "Quit Kunkun", + // function: async () => { + // exit(0) + // } + // }, + // { + // name: "Toggle Dev Extension Live Load Mode", + // iconifyIcon: "ri:toggle-line", + // description: "Load dev extensions from their dev server URLs", + // function: async () => { + // toggleDevExtensionLiveLoadMode() + // } + // }, + // { + // name: "Toggle Hide On Blur", + // iconifyIcon: "ri:toggle-line", + // description: "Toggle Hide On Blur", + // function: async () => { + // const appConfig = useAppConfigStore() + // appConfig.setHideOnBlur(!appConfig.hideOnBlur) + // const appStateStore = useAppStateStore() + // appStateStore.setSearchTermSync("") + // toast.success(`"Hide on Blur" toggled to: ${appConfig.hideOnBlur}`) + // } + // } +] diff --git a/apps/desktop/src/lib/cmds/ext.ts b/apps/desktop/src/lib/cmds/ext.ts new file mode 100644 index 0000000..5470804 --- /dev/null +++ b/apps/desktop/src/lib/cmds/ext.ts @@ -0,0 +1,57 @@ +import { winExtMap } from "@/stores/winExtMap" +import { trimSlash } from "@/utils/url" +import { constructExtensionSupportDir } from "@kksh/api" +import { CmdTypeEnum, CustomUiCmd, ExtPackageJsonExtra, TemplateUiCmd } from "@kksh/api/models" +import { launchNewExtWindow } from "@kksh/extension" +import { convertFileSrc } from "@tauri-apps/api/core" +import { WebviewWindow } from "@tauri-apps/api/webviewWindow" +import * as fs from "@tauri-apps/plugin-fs" +import { debug } from "@tauri-apps/plugin-log" +import { goto } from "$app/navigation" +import * as v from "valibot" + +export async function createExtSupportDir(extPath: string) { + const extSupportDir = await constructExtensionSupportDir(extPath) + if (!(await fs.exists(extSupportDir))) { + await fs.mkdir(extSupportDir, { recursive: true }) + } +} + +export async function onTemplateUiCmdSelect( + ext: ExtPackageJsonExtra, + cmd: TemplateUiCmd, + { isDev, hmr }: { isDev: boolean; hmr: boolean } +) { + await createExtSupportDir(ext.extPath) + console.log("onTemplateUiCmdSelect", ext, cmd, isDev, hmr) +} + +export async function onCustomUiCmdSelect( + ext: ExtPackageJsonExtra, + cmd: CustomUiCmd, + { isDev, hmr }: { isDev: boolean; hmr: boolean } +) { + console.log("onCustomUiCmdSelect", ext, cmd, isDev, hmr) + await createExtSupportDir(ext.extPath) + let url = cmd.main + + if (hmr && isDev && cmd.devMain) { + url = cmd.devMain + } else { + url = decodeURIComponent(convertFileSrc(`${trimSlash(cmd.main)}`, "ext")) + } + const url2 = `/extension/ui-iframe?url=${encodeURIComponent(url)}&extPath=${encodeURIComponent(ext.extPath)}` + if (cmd.window) { + const winLabel = await winExtMap.registerExtensionWithWindow({ + extPath: ext.extPath, + dist: cmd.dist + }) + console.log("Launch new window, ", winLabel) + const window = launchNewExtWindow(winLabel, url2, cmd.window) + } else { + console.log("Launch main window") + return winExtMap + .registerExtensionWithWindow({ windowLabel: "main", extPath: ext.extPath, dist: cmd.dist }) + .then(() => goto(url2)) + } +} diff --git a/apps/desktop/src/lib/cmds/index.ts b/apps/desktop/src/lib/cmds/index.ts new file mode 100644 index 0000000..3efc68d --- /dev/null +++ b/apps/desktop/src/lib/cmds/index.ts @@ -0,0 +1,23 @@ +import { CmdTypeEnum, CustomUiCmd, ExtPackageJsonExtra, TemplateUiCmd } from "@kksh/api/models" +import type { CommandLaunchers, OnExtCmdSelect } from "@kksh/ui/types" +import * as v from "valibot" +import { onCustomUiCmdSelect, onTemplateUiCmdSelect } from "./ext" + +const onExtCmdSelect: OnExtCmdSelect = ( + ext: ExtPackageJsonExtra, + cmd: CustomUiCmd | TemplateUiCmd, + { isDev, hmr }: { isDev: boolean; hmr: boolean } +) => { + switch (cmd.type) { + case CmdTypeEnum.UiIframe: + onCustomUiCmdSelect(ext, v.parse(CustomUiCmd, cmd), { isDev, hmr }) + break + case CmdTypeEnum.UiWorker: + onTemplateUiCmdSelect(ext, v.parse(TemplateUiCmd, cmd), { isDev, hmr }) + break + default: + console.error("Unknown command type", cmd.type) + } +} + +export const commandLaunchers = { onExtCmdSelect } satisfies CommandLaunchers diff --git a/apps/desktop/src/lib/components/context/AppContext.svelte b/apps/desktop/src/lib/components/context/AppContext.svelte new file mode 100644 index 0000000..11ac3e8 --- /dev/null +++ b/apps/desktop/src/lib/components/context/AppContext.svelte @@ -0,0 +1,19 @@ + + +{@render children?.()} diff --git a/apps/desktop/src/lib/components/main/CommandPalette.svelte b/apps/desktop/src/lib/components/main/CommandPalette.svelte new file mode 100644 index 0000000..f608c2a --- /dev/null +++ b/apps/desktop/src/lib/components/main/CommandPalette.svelte @@ -0,0 +1,75 @@ + + + + + + + + No results found. + + {#if $appConfig.extensionPath} + + isExtPathInDev($appConfig.extensionPath!, ext.extPath) + )} + heading="Dev Extensions" + isDev={true} + onExtCmdSelect={commandLaunchers.onExtCmdSelect} + hmr={$appConfig.hmr} + /> + {/if} + {#if $appConfig.extensionPath} + !isExtPathInDev($appConfig.extensionPath!, ext.extPath) + )} + heading="Extensions" + isDev={false} + hmr={false} + onExtCmdSelect={commandLaunchers.onExtCmdSelect} + /> + {/if} + + + + diff --git a/apps/desktop/src/lib/components/standalone/settings/DevExtPathForm.svelte b/apps/desktop/src/lib/components/standalone/settings/DevExtPathForm.svelte new file mode 100644 index 0000000..2860652 --- /dev/null +++ b/apps/desktop/src/lib/components/standalone/settings/DevExtPathForm.svelte @@ -0,0 +1,43 @@ + + +
+ + + +
diff --git a/apps/desktop/src/lib/constants.ts b/apps/desktop/src/lib/constants.ts new file mode 100644 index 0000000..bd095c2 --- /dev/null +++ b/apps/desktop/src/lib/constants.ts @@ -0,0 +1,20 @@ +import { appIsDev } from "@kksh/api/commands" +import { appDataDir, join } from "@tauri-apps/api/path" +import * as fs from "@tauri-apps/plugin-fs" +import { PUBLIC_SUPABASE_ANON_KEY, PUBLIC_SUPABASE_PROJECT_ID } from "$env/static/public" + +export const SUPABASE_ANON_KEY = PUBLIC_SUPABASE_ANON_KEY +export const SUPABASE_URL = `https://${PUBLIC_SUPABASE_PROJECT_ID}.supabase.co` +export const SUPABASE_GRAPHQL_ENDPOINT = `${SUPABASE_URL}/graphql/v1` +export function getExtensionsFolder() { + return appDataDir() + .then((appDataDirPath) => join(appDataDirPath, "extensions")) + .then(async (path) => { + if (!(await fs.exists(path))) { + await fs.mkdir(path) + } + return path + }) +} +export const IS_IN_TAURI = + typeof window !== "undefined" && (window as any).__TAURI_INTERNALS__ !== undefined diff --git a/apps/desktop/src/lib/context/appConfig.ts b/apps/desktop/src/lib/context/appConfig.ts new file mode 100644 index 0000000..4537631 --- /dev/null +++ b/apps/desktop/src/lib/context/appConfig.ts @@ -0,0 +1,18 @@ +/** + * This is app state context. + * It's designed to allow all components to access a shared state. + * With context, we can avoid prop drilling, and avoid using stores which makes components hard to encapsulate. + */ +import type { AppConfig } from "@/types/appConfig" +import { getContext, setContext } from "svelte" +import type { Writable } from "svelte/store" + +export const APP_CONFIG_CONTEXT_KEY = Symbol("appConfig") + +export function getAppConfigContext(): Writable { + return getContext(APP_CONFIG_CONTEXT_KEY) +} + +export function setAppConfigContext(appConfig: Writable) { + setContext(APP_CONFIG_CONTEXT_KEY, appConfig) +} diff --git a/apps/desktop/src/lib/context/appState.ts b/apps/desktop/src/lib/context/appState.ts new file mode 100644 index 0000000..12aa792 --- /dev/null +++ b/apps/desktop/src/lib/context/appState.ts @@ -0,0 +1,13 @@ +import type { AppState } from "@/types/appState" +import { getContext, setContext } from "svelte" +import type { Writable } from "svelte/store" + +export const APP_STATE_CONTEXT_KEY = Symbol("appState") + +export function getAppStateContext(): Writable { + return getContext(APP_STATE_CONTEXT_KEY) +} + +export function setAppStateContext(appState: Writable) { + setContext(APP_STATE_CONTEXT_KEY, appState) +} diff --git a/apps/desktop/src/lib/context/index.ts b/apps/desktop/src/lib/context/index.ts new file mode 100644 index 0000000..46db929 --- /dev/null +++ b/apps/desktop/src/lib/context/index.ts @@ -0,0 +1 @@ +export * from "./appConfig" diff --git a/apps/desktop/src/lib/stores/appConfig.ts b/apps/desktop/src/lib/stores/appConfig.ts new file mode 100644 index 0000000..aa50558 --- /dev/null +++ b/apps/desktop/src/lib/stores/appConfig.ts @@ -0,0 +1,86 @@ +import { getExtensionsFolder } from "@/constants" +import { themeConfigStore, updateTheme, type ThemeConfig } from "@kksh/svelte5" +import { PersistedAppConfig, type AppConfig } from "@kksh/types" +import * as path from "@tauri-apps/api/path" +import { remove } from "@tauri-apps/plugin-fs" +import { debug, error } from "@tauri-apps/plugin-log" +import * as os from "@tauri-apps/plugin-os" +import { load } from "@tauri-apps/plugin-store" +import { get, writable, type Writable } from "svelte/store" +import * as v from "valibot" + +export const defaultAppConfig: AppConfig = { + isInitialized: false, + platform: "macos", + theme: { + theme: "zinc", + radius: 0.5, + lightMode: "auto" + }, + triggerHotkey: null, + launchAtLogin: true, + showInTray: true, + devExtensionPath: null, + extensionPath: undefined, + hmr: false, + hideOnBlur: true, + extensionAutoUpgrade: true, + joinBetaProgram: false, + onBoarded: false +} + +interface AppConfigAPI { + init: () => Promise + setTheme: (theme: ThemeConfig) => void + setDevExtensionPath: (devExtensionPath: string | null) => void +} + +function createAppConfig(): Writable & AppConfigAPI { + const { subscribe, update, set } = writable(defaultAppConfig) + + async function init() { + debug("Initializing app config") + const appDataDir = await path.appDataDir() + // const appConfigPath = await path.join(appDataDir, "appConfig.json") + // debug(`appConfigPath: ${appConfigPath}`) + const persistStore = await load("kk-config.json", { autoSave: true }) + const loadedConfig = await persistStore.get("config") + const parseRes = v.safeParse(PersistedAppConfig, loadedConfig) + if (parseRes.success) { + console.log("Parse Persisted App Config Success", parseRes.output) + const extensionPath = await path.join(appDataDir, "extensions") + update((config) => ({ + ...config, + ...parseRes.output, + isInitialized: true, + extensionPath, + platform: os.platform() + })) + } else { + error("Failed to parse app config, going to remove it and reinitialize") + console.error(v.flatten(parseRes.issues)) + await persistStore.clear() + await persistStore.set("config", v.parse(PersistedAppConfig, defaultAppConfig)) + } + + subscribe(async (config) => { + console.log("Saving app config", config) + await persistStore.set("config", config) + updateTheme(config.theme) + }) + } + + return { + setTheme: (theme: ThemeConfig) => update((config) => ({ ...config, theme })), + setDevExtensionPath: (devExtensionPath: string | null) => { + console.log("setDevExtensionPath", devExtensionPath) + update((config) => ({ ...config, devExtensionPath })) + }, + init, + subscribe, + update, + set + } +} + +export const appConfig = createAppConfig() diff --git a/apps/desktop/src/lib/stores/appState.ts b/apps/desktop/src/lib/stores/appState.ts new file mode 100644 index 0000000..aaaf45d --- /dev/null +++ b/apps/desktop/src/lib/stores/appState.ts @@ -0,0 +1,28 @@ +import type { AppState } from "@/types" +import { get, writable, type Writable } from "svelte/store" + +export const defaultAppState: AppState = { + searchTerm: "", + highlightedCmd: "" +} + +interface AppStateAPI { + clearSearchTerm: () => void + get: () => AppState +} + +function createAppState(): Writable & AppStateAPI { + const store = writable(defaultAppState) + + return { + subscribe: store.subscribe, + update: store.update, + set: store.set, + get: () => get(store), + clearSearchTerm: () => { + store.update((state) => ({ ...state, searchTerm: "" })) + } + } +} + +export const appState = createAppState() diff --git a/apps/desktop/src/lib/stores/cmds.ts b/apps/desktop/src/lib/stores/cmds.ts new file mode 100644 index 0000000..a30b1e5 --- /dev/null +++ b/apps/desktop/src/lib/stores/cmds.ts @@ -0,0 +1 @@ +import { derived, fromStore, get, readable, readonly, toStore, writable } from "svelte/store" diff --git a/apps/desktop/src/lib/stores/extensions.ts b/apps/desktop/src/lib/stores/extensions.ts new file mode 100644 index 0000000..00b5bc6 --- /dev/null +++ b/apps/desktop/src/lib/stores/extensions.ts @@ -0,0 +1,124 @@ +import { getExtensionsFolder } from "@/constants" +import { db } from "@kksh/api/commands" +import type { ExtPackageJsonExtra } from "@kksh/api/models" +import * as extAPI from "@kksh/extension" +import * as path from "@tauri-apps/api/path" +import * as fs from "@tauri-apps/plugin-fs" +import { derived, get, writable, type Readable, type Writable } from "svelte/store" +import { appConfig } from "./appConfig" + +function createExtensionsStore(): Writable & { + init: () => Promise + getExtensionsFromStore: () => ExtPackageJsonExtra[] + installFromTarballUrl: (tarballUrl: string, installDir: string) => Promise + findStoreExtensionByIdentifier: (identifier: string) => ExtPackageJsonExtra | undefined + registerNewExtensionByPath: (extPath: string) => Promise + uninstallStoreExtensionByIdentifier: (identifier: string) => Promise + upgradeStoreExtension: (identifier: string, tarballUrl: string) => Promise +} { + const { subscribe, update, set } = writable([]) + + function init() { + return extAPI.loadAllExtensionsFromDb().then((exts) => { + set(exts) + }) + } + + function getExtensionsFromStore() { + const extContainerPath = get(appConfig).extensionPath + if (!extContainerPath) return [] + return get(extensions).filter((ext) => !extAPI.isExtPathInDev(extContainerPath, ext.extPath)) + } + + function findStoreExtensionByIdentifier(identifier: string) { + return get(extensions).find((ext) => ext.kunkun.identifier === identifier) + } + + /** + * After install, register the extension to the store + * @param extPath absolute path to the extension folder + * @returns loaded extension + */ + async function registerNewExtensionByPath(extPath: string) { + return extAPI + .loadExtensionManifestFromDisk(await path.join(extPath, "package.json")) + .then((ext) => { + update((exts) => { + const existingExt = exts.find((e) => e.extPath === ext.extPath) + if (existingExt) return exts + return [...exts, ext] + }) + return ext + }) + .catch((err) => { + console.error(err) + return Promise.reject(err) + }) + } + + async function installFromTarballUrl(tarballUrl: string, extsDir: string) { + return extAPI.installTarballUrl(tarballUrl, extsDir).then((extInstallPath) => { + return registerNewExtensionByPath(extInstallPath) + }) + } + + async function uninstallExtensionByPath(targetPath: string) { + const targetExt = get(extensions).find((ext) => ext.extPath === targetPath) + if (!targetExt) throw new Error(`Extension ${targetPath} not registered in DB`) + console.log(extAPI) + + return extAPI + .uninstallExtensionByPath(targetPath) + .then(() => update((exts) => exts.filter((ext) => ext.extPath !== targetExt.extPath))) + .then(() => targetExt) + } + + async function uninstallStoreExtensionByIdentifier(identifier: string) { + const targetExt = getExtensionsFromStore().find((ext) => ext.kunkun.identifier === identifier) + if (!targetExt) throw new Error(`Extension ${identifier} not found`) + return uninstallExtensionByPath(targetExt.extPath) + } + + async function upgradeStoreExtension( + identifier: string, + tarballUrl: string + ): Promise { + const extsDir = get(appConfig).extensionPath + if (!extsDir) throw new Error("Extension path not set") + return uninstallStoreExtensionByIdentifier(identifier).then(() => + installFromTarballUrl(tarballUrl, extsDir) + ) + } + + return { + init, + getExtensionsFromStore, + findStoreExtensionByIdentifier, + registerNewExtensionByPath, + installFromTarballUrl, + uninstallStoreExtensionByIdentifier, + upgradeStoreExtension, + subscribe, + update, + set + } +} + +export const extensions = createExtensionsStore() + +// export const devExtensions: Readable = derived( +// extensions, +// ($extensionsStore, set) => { +// getExtensionsFolder().then((extFolder) => { +// set($extensionsStore.filter((ext) => !ext.extPath.startsWith(extFolder))) +// }) +// } +// ) +export const installedStoreExts: Readable = derived( + extensions, + ($extensionsStore) => { + const extContainerPath = get(appConfig).extensionPath + if (!extContainerPath) return [] + return $extensionsStore.filter((ext) => !extAPI.isExtPathInDev(extContainerPath, ext.extPath)) + } +) diff --git a/apps/desktop/src/lib/stores/index.ts b/apps/desktop/src/lib/stores/index.ts new file mode 100644 index 0000000..daaa348 --- /dev/null +++ b/apps/desktop/src/lib/stores/index.ts @@ -0,0 +1,4 @@ +export * from "./appConfig" +export * from "./appState" +export * from "./winExtMap" +export * from "./extensions" diff --git a/apps/desktop/src/lib/stores/winExtMap.ts b/apps/desktop/src/lib/stores/winExtMap.ts new file mode 100644 index 0000000..7b5cd5f --- /dev/null +++ b/apps/desktop/src/lib/stores/winExtMap.ts @@ -0,0 +1,119 @@ +/** + * Store in this file is used to map window labels to extension paths and pids + * The purpose is to keep track of which extensions are running in which windows, and the left over processes when the extension is closed + */ +import { killProcesses } from "@/utils/process" +import { + getExtLabelMap, + registerExtensionSpawnedProcess, + registerExtensionWindow, + unregisterExtensionSpawnedProcess, + unregisterExtensionWindow +} from "@kksh/api/commands" +import { warn } from "@tauri-apps/plugin-log" +import { get, writable, type Writable } from "svelte/store" + +export type WinExtMap = Record< + string, + { + windowLabel: string + extPath: string + pids: number[] + } +> + +type API = { + init: () => Promise + registerExtensionWithWindow: (options: { + windowLabel?: string + extPath: string + dist?: string + }) => Promise + unregisterExtensionFromWindow: (windowLabel: string) => Promise + registerProcess: (windowLabel: string, pid: number) => Promise + unregisterProcess: (pid: number) => Promise +} + +function createWinExtMapStore(): Writable & API { + const store = writable({}) + + async function init() {} + + return { + init, + registerExtensionWithWindow: async ({ + extPath, + windowLabel, + dist + }: { + extPath: string + windowLabel?: string + dist?: string + }) => { + const winExtMap = get(store) + if (windowLabel) { + if (winExtMap[windowLabel]) { + // there is a previous extension registered in this window but not cleaned up properly + warn(`Window ${windowLabel} has a previous extension registered but not cleaned up`) + await killProcesses(winExtMap[windowLabel].pids) + delete winExtMap[windowLabel] + } else { + winExtMap[windowLabel] = { + windowLabel, + extPath, + pids: [] + } + } + } + const returnedWinLabel = await registerExtensionWindow({ + extensionPath: extPath, + windowLabel, + dist + }) + store.set(winExtMap) + return returnedWinLabel + }, + unregisterExtensionFromWindow: async (windowLabel: string) => { + const winExtMap = get(store) + if (winExtMap[windowLabel]) { + // clean up processes spawned by extension but not killed by itself + const extLabelMap = await getExtLabelMap() // realtime data from core process + Object.entries(extLabelMap).forEach(([label, ext]) => { + if (label === windowLabel) { + killProcesses(ext.processes) + } + }) + await unregisterExtensionWindow(windowLabel) + delete winExtMap[windowLabel] + store.set(winExtMap) + } else { + warn(`Window ${windowLabel} does not have an extension registered`) + } + }, + registerProcess: async (windowLabel: string, pid: number) => { + const winExtMap = get(store) + registerExtensionSpawnedProcess(windowLabel, pid) + if (!winExtMap[windowLabel]) { + throw new Error(`Window ${windowLabel} does not have an extension registered`) + } + winExtMap[windowLabel].pids.push(pid) + store.set(winExtMap) + }, + unregisterProcess: async (pid: number) => { + const winExtMap = get(store) + const found = Object.entries(winExtMap).find(([windowLabel, ext]) => ext.pids.includes(pid)) + if (!found) { + return + } + const [windowLabel, ext] = found + return unregisterExtensionSpawnedProcess(windowLabel, pid).then(() => { + ext.pids = ext.pids.filter((p) => p !== pid) + }) + }, + subscribe: store.subscribe, + update: store.update, + set: store.set + } +} + +export const winExtMap = createWinExtMapStore() diff --git a/apps/desktop/src/lib/supabase.ts b/apps/desktop/src/lib/supabase.ts new file mode 100644 index 0000000..a9927cd --- /dev/null +++ b/apps/desktop/src/lib/supabase.ts @@ -0,0 +1,7 @@ +import { createSB, SupabaseAPI } from "@kksh/supabase" +import { SUPABASE_ANON_KEY, SUPABASE_URL } from "./constants" + +export const supabase = createSB(SUPABASE_URL, SUPABASE_ANON_KEY) +export const storage = supabase.storage +export const supabaseExtensionsStorage = supabase.storage.from("extensions") +export const supabaseAPI = new SupabaseAPI(supabase) diff --git a/packages/ui/index.ts b/apps/desktop/src/lib/types/index.ts similarity index 100% rename from packages/ui/index.ts rename to apps/desktop/src/lib/types/index.ts diff --git a/apps/desktop/src/lib/utils.ts b/apps/desktop/src/lib/utils.ts deleted file mode 100644 index ac680b3..0000000 --- a/apps/desktop/src/lib/utils.ts +++ /dev/null @@ -1,6 +0,0 @@ -import { type ClassValue, clsx } from "clsx"; -import { twMerge } from "tailwind-merge"; - -export function cn(...inputs: ClassValue[]) { - return twMerge(clsx(inputs)); -} diff --git a/apps/desktop/src/lib/utils/key.ts b/apps/desktop/src/lib/utils/key.ts new file mode 100644 index 0000000..62f16f1 --- /dev/null +++ b/apps/desktop/src/lib/utils/key.ts @@ -0,0 +1,25 @@ +import { appState } from "@/stores" +import { goto } from "$app/navigation" +import { goBack, goHome } from "./route" + +export function goHomeOnEscape(e: KeyboardEvent) { + if (e.key === "Escape") { + goHome() + } +} + +export function goBackOnEscape(e: KeyboardEvent) { + if (e.key === "Escape") { + goBack() + } +} + +export function goBackOnEscapeClearSearchTerm(e: KeyboardEvent) { + if (e.key === "Escape") { + if (appState.get().searchTerm) { + appState.clearSearchTerm() + } else { + goBack() + } + } +} diff --git a/apps/desktop/src/lib/utils/process.ts b/apps/desktop/src/lib/utils/process.ts new file mode 100644 index 0000000..a4d1921 --- /dev/null +++ b/apps/desktop/src/lib/utils/process.ts @@ -0,0 +1,13 @@ +import { error } from "@tauri-apps/plugin-log" +import { Child } from "tauri-plugin-shellx-api" + +export function killProcesses(pids: number[]) { + return Promise.all( + pids.map((pid) => { + return new Child(pid).kill().catch((err) => { + error(`Failed to kill process ${pid}, ${err}`) + console.error(`Failed to kill process ${pid}`, err) + }) + }) + ) +} diff --git a/apps/desktop/src/lib/utils/route.ts b/apps/desktop/src/lib/utils/route.ts new file mode 100644 index 0000000..e483218 --- /dev/null +++ b/apps/desktop/src/lib/utils/route.ts @@ -0,0 +1,9 @@ +import { goto } from "$app/navigation" + +export function goBack() { + window.history.back() +} + +export function goHome() { + goto("/") +} diff --git a/apps/desktop/src/lib/utils/style.ts b/apps/desktop/src/lib/utils/style.ts new file mode 100644 index 0000000..736d2eb --- /dev/null +++ b/apps/desktop/src/lib/utils/style.ts @@ -0,0 +1,16 @@ +import { type Position } from "@kksh/api/models" + +export function positionToTailwindClasses(position: Position) { + switch (position) { + case "top-left": + return "top-2 left-2" + case "top-right": + return "top-2 right-2" + case "bottom-left": + return "bottom-2 left-2" + case "bottom-right": + return "bottom-2 right-2" + default: + return "" + } +} diff --git a/apps/desktop/src/lib/utils/url.ts b/apps/desktop/src/lib/utils/url.ts new file mode 100644 index 0000000..472bb2a --- /dev/null +++ b/apps/desktop/src/lib/utils/url.ts @@ -0,0 +1,3 @@ +export function trimSlash(str: string) { + return str.replace(/^\/+|\/+$/g, "") +} diff --git a/apps/desktop/src/lib/utils/window.ts b/apps/desktop/src/lib/utils/window.ts new file mode 100644 index 0000000..315209c --- /dev/null +++ b/apps/desktop/src/lib/utils/window.ts @@ -0,0 +1,5 @@ +import { getCurrentWindow } from "@tauri-apps/api/window" + +export function isInMainWindow() { + return getCurrentWindow().label == "main" +} diff --git a/apps/desktop/src/routes/+error.svelte b/apps/desktop/src/routes/+error.svelte new file mode 100644 index 0000000..91a5da3 --- /dev/null +++ b/apps/desktop/src/routes/+error.svelte @@ -0,0 +1,22 @@ + + + + + + window.history.back()} + rawJsonError={JSON.stringify($page, null, 2)} + /> + diff --git a/apps/desktop/src/routes/+layout.svelte b/apps/desktop/src/routes/+layout.svelte index d12dabc..99c8d80 100644 --- a/apps/desktop/src/routes/+layout.svelte +++ b/apps/desktop/src/routes/+layout.svelte @@ -1,9 +1,41 @@ + -{@render children()} + + + {@render children()} + + diff --git a/apps/desktop/src/routes/+layout.ts b/apps/desktop/src/routes/+layout.ts index f4fb689..8d7e6d2 100644 --- a/apps/desktop/src/routes/+layout.ts +++ b/apps/desktop/src/routes/+layout.ts @@ -1,5 +1,5 @@ // Tauri doesn't have a Node.js server to do proper SSR // so we will use adapter-static to prerender the app (SSG) // See: https://v2.tauri.app/start/frontend/sveltekit/ for more info -export const prerender = true; -export const ssr = false; +export const prerender = true +export const ssr = false diff --git a/apps/desktop/src/routes/+page.svelte b/apps/desktop/src/routes/+page.svelte index 1f8d7b3..3f4ab05 100644 --- a/apps/desktop/src/routes/+page.svelte +++ b/apps/desktop/src/routes/+page.svelte @@ -1,14 +1,18 @@ - - - - Heads up! - You can add components to your app using the cli. - - - - CN - + diff --git a/apps/desktop/src/routes/extension/store/+page.svelte b/apps/desktop/src/routes/extension/store/+page.svelte new file mode 100644 index 0000000..97e9379 --- /dev/null +++ b/apps/desktop/src/routes/extension/store/+page.svelte @@ -0,0 +1,77 @@ + + + + diff --git a/apps/desktop/src/routes/extension/store/+page.ts b/apps/desktop/src/routes/extension/store/+page.ts new file mode 100644 index 0000000..d002751 --- /dev/null +++ b/apps/desktop/src/routes/extension/store/+page.ts @@ -0,0 +1,39 @@ +import { appConfig, extensions, installedStoreExts } from "@/stores" +import { supabaseAPI } from "@/supabase" +import type { ExtPackageJsonExtra } from "@kksh/api/models" +import { SBExt } from "@kksh/api/supabase" +import { isExtPathInDev, isUpgradable } from "@kksh/extension" +import { error } from "@sveltejs/kit" +import { derived, get, type Readable } from "svelte/store" +import type { PageLoad } from "./$types" + +export const load: PageLoad = async (): Promise<{ + storeExtList: SBExt[] + installedStoreExts: Readable + installedExtsMap: Readable> + upgradableExpsMap: Readable> +}> => { + const storeExtList = await supabaseAPI.getExtList() + // map identifier to extItem + const storeExtsMap = Object.fromEntries(storeExtList.map((ext) => [ext.identifier, ext])) + const _appConfig = get(appConfig) + // const installedStoreExts = derived(extensions, ($extensions) => { + // if (!_appConfig.extensionPath) return [] + // return $extensions.filter((ext) => !isExtPathInDev(_appConfig.extensionPath!, ext.extPath)) + // }) + // map installed extension identifier to version + const installedExtsMap = derived(installedStoreExts, ($exts) => + Object.fromEntries($exts.map((ext) => [ext.kunkun.identifier, ext.version])) + ) + const upgradableExpsMap = derived(installedStoreExts, ($exts) => + Object.fromEntries( + $exts.map((ext) => { + const dbExt: SBExt | undefined = storeExtsMap[ext.kunkun.identifier] + return [ext.kunkun.identifier, dbExt ? isUpgradable(dbExt, ext.version) : false] + }) + ) + ) + console.log(get(upgradableExpsMap)) + + return { storeExtList, installedStoreExts, installedExtsMap, upgradableExpsMap } +} diff --git a/apps/desktop/src/routes/extension/store/[identifier]/+error.svelte b/apps/desktop/src/routes/extension/store/[identifier]/+error.svelte new file mode 100644 index 0000000..4768785 --- /dev/null +++ b/apps/desktop/src/routes/extension/store/[identifier]/+error.svelte @@ -0,0 +1,23 @@ + + + + + + goto("/")} + rawJsonError={JSON.stringify($page, null, 2)} + /> + diff --git a/apps/desktop/src/routes/extension/store/[identifier]/+page.svelte b/apps/desktop/src/routes/extension/store/[identifier]/+page.svelte new file mode 100644 index 0000000..78ed467 --- /dev/null +++ b/apps/desktop/src/routes/extension/store/[identifier]/+page.svelte @@ -0,0 +1,120 @@ + + + + + + diff --git a/apps/desktop/src/routes/extension/store/[identifier]/+page.ts b/apps/desktop/src/routes/extension/store/[identifier]/+page.ts new file mode 100644 index 0000000..c79c946 --- /dev/null +++ b/apps/desktop/src/routes/extension/store/[identifier]/+page.ts @@ -0,0 +1,38 @@ +import { extensions } from "@/stores" +import { supabaseAPI } from "@/supabase" +import { KunkunExtManifest, type ExtPackageJsonExtra } from "@kksh/api/models" +import type { Tables } from "@kksh/api/supabase/types" +import { error } from "@sveltejs/kit" +import { toast } from "svelte-sonner" +import { get } from "svelte/store" +import * as v from "valibot" +import type { PageLoad } from "./$types" + +export const load: PageLoad = async ({ + params +}): Promise<{ + ext: Tables<"ext_publish"> + manifest: KunkunExtManifest +}> => { + const { error: dbError, data: ext } = await supabaseAPI.getLatestExtPublish(params.identifier) + if (dbError) { + return error(400, { + message: dbError.message + }) + } + + const parseManifest = v.safeParse(KunkunExtManifest, ext.manifest) + if (!parseManifest.success) { + const errMsg = "Invalid extension manifest, you may need to upgrade your app." + toast.error(errMsg) + throw error(400, errMsg) + } + + return { + ext, + manifest: parseManifest.output + } +} + +export const csr = true +export const prerender = false diff --git a/apps/desktop/src/routes/extension/ui-iframe/+page.svelte b/apps/desktop/src/routes/extension/ui-iframe/+page.svelte new file mode 100644 index 0000000..ec59ab7 --- /dev/null +++ b/apps/desktop/src/routes/extension/ui-iframe/+page.svelte @@ -0,0 +1,202 @@ + + + + +{#if uiControl.backBtnPosition} + +{/if} +{#if uiControl.moveBtnPosition} + +{/if} +{#if uiControl.refreshBtnPosition} + +{/if} + +
+ +
diff --git a/apps/desktop/src/routes/extension/ui-iframe/+page.ts b/apps/desktop/src/routes/extension/ui-iframe/+page.ts new file mode 100644 index 0000000..94efd0c --- /dev/null +++ b/apps/desktop/src/routes/extension/ui-iframe/+page.ts @@ -0,0 +1,50 @@ +import { db, unregisterExtensionWindow } from "@kksh/api/commands" +import type { Ext as ExtInfoInDB, ExtPackageJsonExtra } from "@kksh/api/models" +import { loadExtensionManifestFromDisk } from "@kksh/extension" +import { join } from "@tauri-apps/api/path" +import { error } from "@tauri-apps/plugin-log" +import { goto } from "$app/navigation" +import { toast } from "svelte-sonner" +import { z } from "zod" +import type { PageLoad } from "./$types" + +export const load: PageLoad = async ({ + url, + params, + route +}): Promise<{ + extPath: string + url: string + loadedExt: ExtPackageJsonExtra + extInfoInDB: ExtInfoInDB +}> => { + // both query parameter must exist + const _extPath = url.searchParams.get("extPath") + const _extUrl = url.searchParams.get("url") + if (!_extPath || !_extUrl) { + toast.error("Invalid extension path or url") + error("Invalid extension path or url") + goto("/") + } + const extPath = z.string().parse(_extPath) + const extUrl = z.string().parse(_extUrl) + let _loadedExt: ExtPackageJsonExtra | undefined + try { + _loadedExt = await loadExtensionManifestFromDisk(await join(extPath, "package.json")) + } catch (err) { + error(`Error loading extension manifest: ${err}`) + toast.error("Error loading extension manifest", { + description: `${err}` + }) + goto("/") + } + const loadedExt = _loadedExt! + const extInfoInDB = await db.getUniqueExtensionByPath(loadedExt.extPath) + if (!extInfoInDB) { + toast.error("Unexpected Error", { + description: `Extension ${loadedExt.kunkun.identifier} not found in database. Run Troubleshooter.` + }) + goto("/") + } + return { extPath, url: extUrl, loadedExt, extInfoInDB: extInfoInDB! } +} diff --git a/apps/desktop/src/routes/extension/ui-worker/+page.svelte b/apps/desktop/src/routes/extension/ui-worker/+page.svelte new file mode 100644 index 0000000..5bf03f4 --- /dev/null +++ b/apps/desktop/src/routes/extension/ui-worker/+page.svelte @@ -0,0 +1 @@ + diff --git a/apps/desktop/src/routes/settings/set-dev-ext-path/+page.svelte b/apps/desktop/src/routes/settings/set-dev-ext-path/+page.svelte new file mode 100644 index 0000000..867100e --- /dev/null +++ b/apps/desktop/src/routes/settings/set-dev-ext-path/+page.svelte @@ -0,0 +1,18 @@ + + + + +
+
+

Set Dev Extension Path

+

This is where your extensions will be installed.

+ +
diff --git a/apps/desktop/static/favicon.png b/apps/desktop/static/favicon.png index 825b9e6..0a514e6 100644 Binary files a/apps/desktop/static/favicon.png and b/apps/desktop/static/favicon.png differ diff --git a/apps/desktop/static/svelte.svg b/apps/desktop/static/svelte.svg deleted file mode 100644 index c5e0848..0000000 --- a/apps/desktop/static/svelte.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/apps/desktop/static/tauri.svg b/apps/desktop/static/tauri.svg deleted file mode 100644 index 31b62c9..0000000 --- a/apps/desktop/static/tauri.svg +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - diff --git a/apps/desktop/static/vite.svg b/apps/desktop/static/vite.svg deleted file mode 100644 index e7b8dfb..0000000 --- a/apps/desktop/static/vite.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/apps/desktop/svelte.config.js b/apps/desktop/svelte.config.js index 96949b1..b626a71 100644 --- a/apps/desktop/svelte.config.js +++ b/apps/desktop/svelte.config.js @@ -1,20 +1,23 @@ // Tauri doesn't have a Node.js server to do proper SSR // so we will use adapter-static to prerender the app (SSG) // See: https://v2.tauri.app/start/frontend/sveltekit/ for more info -import adapter from "@sveltejs/adapter-static"; -import { vitePreprocess } from "@sveltejs/vite-plugin-svelte"; +import adapter from "@sveltejs/adapter-static" +import { vitePreprocess } from "@sveltejs/vite-plugin-svelte" /** @type {import('@sveltejs/kit').Config} */ const config = { - preprocess: vitePreprocess(), - kit: { - adapter: adapter(), - alias: { - "@/*": "./src/lib/*", - "@kksh/ui/*": "../../packages/ui/*", - "@kksh/svelte5/*": "../../node_modules/@kksh/svelte5/src/lib/*", - }, - }, -}; + preprocess: vitePreprocess(), + kit: { + adapter: adapter({ + fallback: "400.html" + // fallback: "index.html" + }), + alias: { + "@/*": "./src/lib/*", + // "@kksh/ui/*": "../../packages/ui/*", + "@kksh/svelte5/*": "../../node_modules/@kksh/svelte5/src/lib/*" + } + } +} -export default config; +export default config diff --git a/package.json b/package.json index eca337e..c980dcc 100644 --- a/package.json +++ b/package.json @@ -1,25 +1,54 @@ { - "name": "kunkun", - "private": true, - "scripts": { - "build": "turbo build", - "dev": "turbo dev", - "lint": "turbo lint", - "format": "prettier --write \"**/*.{ts,tsx,md}\"" - }, - "devDependencies": { - "@ianvs/prettier-plugin-sort-imports": "^4.3.1", - "@kksh/svelte5": "0.1.2-beta.3", - "prettier": "^3.2.5", - "prettier-plugin-svelte": "^3.2.7", - "prettier-plugin-tailwindcss": "^0.6.8", - "svelte": "^5.0.0", - "svelte-check": "^4.0.0", - "turbo": "^2.2.3", - "typescript": "5.5.4" - }, - "packageManager": "pnpm@8.15.6", - "engines": { - "node": ">=18" - } + "name": "kunkun", + "private": true, + "scripts": { + "build": "turbo build", + "dev": "turbo dev", + "test": "turbo run test", + "prepare": "turbo run prepare", + "lint": "turbo lint", + "format": "prettier --write \"**/*.{ts,tsx,md,svelte}\"" + }, + "devDependencies": { + "@ianvs/prettier-plugin-sort-imports": "^4.3.1", + "@kksh/api": "workspace:*", + "@kksh/svelte5": "0.1.2-beta.4", + "prettier": "^3.2.5", + "prettier-plugin-svelte": "^3.2.7", + "prettier-plugin-tailwindcss": "^0.6.8", + "svelte": "^5.0.0", + "svelte-check": "^4.0.0", + "turbo": "^2.2.3", + "typescript": "5.5.4" + }, + "packageManager": "pnpm@9.12.3", + "engines": { + "node": ">=22" + }, + "dependencies": { + "@changesets/cli": "^2.27.9", + "@iconify/svelte": "^4.0.2", + "@supabase/supabase-js": "^2.46.1", + "@tauri-apps/api": "^2.0.3", + "@tauri-apps/cli": "^2.0.4", + "@tauri-apps/plugin-deep-link": "^2.0.0", + "@tauri-apps/plugin-dialog": "^2.0.1", + "@tauri-apps/plugin-fs": "^2.0.1", + "@tauri-apps/plugin-global-shortcut": "^2.0.0", + "@tauri-apps/plugin-http": "^2.0.1", + "@tauri-apps/plugin-log": "^2.0.0", + "@tauri-apps/plugin-notification": "^2.0.0", + "@tauri-apps/plugin-os": "^2.0.0", + "@tauri-apps/plugin-process": "2.0.0", + "@tauri-apps/plugin-shell": "^2.0.1", + "@tauri-apps/plugin-store": "^2.1.0", + "@tauri-apps/plugin-updater": "^2.0.0", + "@tauri-apps/plugin-upload": "^2.0.0", + "supabase": "^1.207.9", + "tauri-plugin-network-api": "workspace:*", + "tauri-plugin-shellx-api": "^2.0.11", + "tauri-plugin-system-info-api": "workspace:*", + "valibot": "^0.40.0", + "zod": "^3.23.8" + } } diff --git a/packages/api/.gitignore b/packages/api/.gitignore new file mode 100644 index 0000000..0828c30 --- /dev/null +++ b/packages/api/.gitignore @@ -0,0 +1,178 @@ +# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs + +logs +_.log +npm-debug.log_ +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Caches + +.cache + +# Diagnostic reports (https://nodejs.org/api/report.html) + +report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json + +# Runtime data + +pids +_.pid +_.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover + +lib-cov + +# Coverage directory used by tools like istanbul + +coverage +*.lcov + +# nyc test coverage + +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) + +.grunt + +# Bower dependency directory (https://bower.io/) + +bower_components + +# node-waf configuration + +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) + +build/Release + +# Dependency directories + +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) + +web_modules/ + +# TypeScript cache + +*.tsbuildinfo + +# Optional npm cache directory + +.npm + +# Optional eslint cache + +.eslintcache + +# Optional stylelint cache + +.stylelintcache + +# Microbundle cache + +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history + +.node_repl_history + +# Output of 'npm pack' + +*.tgz + +# Yarn Integrity file + +.yarn-integrity + +# dotenv environment variable files + +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) + +.parcel-cache + +# Next.js build output + +.next +out + +# Nuxt.js build / generate output + +.nuxt +dist + +# Gatsby files + +# Comment in the public line in if your project uses Gatsby and not Next.js + +# https://nextjs.org/blog/next-9-1#public-directory-support + +# public + +# vuepress build output + +.vuepress/dist + +# vuepress v2.x temp and cache directory + +.temp + +# Docusaurus cache and generated files + +.docusaurus + +# Serverless directories + +.serverless/ + +# FuseBox cache + +.fusebox/ + +# DynamoDB Local files + +.dynamodb/ + +# TernJS port file + +.tern-port + +# Stores VSCode versions used for testing VSCode extensions + +.vscode-test + +# yarn v2 + +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# IntelliJ based IDEs +.idea + +# Finder (MacOS) folder config +.DS_Store +docs +stats.html +deno.d.ts diff --git a/packages/api/.npmrc b/packages/api/.npmrc new file mode 100644 index 0000000..41583e3 --- /dev/null +++ b/packages/api/.npmrc @@ -0,0 +1 @@ +@jsr:registry=https://npm.jsr.io diff --git a/packages/api/README.md b/packages/api/README.md new file mode 100644 index 0000000..b87e23d --- /dev/null +++ b/packages/api/README.md @@ -0,0 +1,43 @@ +# @kksh/api + +![NPM Version](https://img.shields.io/npm/v/%40kksh%2Fapi) + +[Kunkun API](https://www.npmjs.com/package/@kksh/api) is an NPM package designed for developers to create extensions for Kunkun. + +`@kksh/api` provides a set of APIs for extensions to interact with Kunkun and System APIs. The APIs include: + +- Clipboard +- Database +- Dialog +- Event +- Fetch +- File System +- Logger +- Network +- Notification +- Open +- OS +- Path +- Shell +- System Info +- System Commands +- Toast +- UI +- etc. + +Read more details in documentation at https://docs.kunkun.sh, +and generated docs at https://docs.api.kunkun.sh/ + +## Dev + +### Dependency Graph + +To detect circular dependencies + +```bash +pnpm madge ./src/ui/worker/index.ts --circular # detect circular dependencies +pnpm dep-tree ./src/ui/worker/index.ts + + +pnpm test # this will detect circular dependencies in all files +``` diff --git a/packages/api/__tests__/verify-package-export.test.ts b/packages/api/__tests__/verify-package-export.test.ts new file mode 100644 index 0000000..af85b2d --- /dev/null +++ b/packages/api/__tests__/verify-package-export.test.ts @@ -0,0 +1,41 @@ +import path from "path" +import { describe, expect, test } from "bun:test" +import madge from "madge" +import * as v from "valibot" +import { exports } from "../package.json" + +const buildEntries: string[] = Object.entries(exports).filter((e) => typeof e === "string") + +describe("Verify Bundled Package", () => { + test("Test Circular Dependency", async () => { + const pkgRoot = path.join(__dirname, "..") + const paths = buildEntries.map((p) => path.join(pkgRoot, p)).map((p) => path.resolve(p)) + // expect each paths to exist + paths.forEach(async (p) => { + expect(await Bun.file(p).exists()).toBe(true) + const madgeRes = await madge(p) + expect(madgeRes.circular()).toEqual([]) + }) + }) + + test("Verify Package Export Path", async () => { + const pkgRoot = path.join(__dirname, "..") + const pkgJsonPath = path.join(pkgRoot, "package.json") + const file = Bun.file(pkgJsonPath) + const pkgJson = await file.json() + const exports = pkgJson["exports"] + Object.entries(exports).forEach(async ([key, value]) => { + const exportPaths = v.parse(v.union([v.record(v.string(), v.string()), v.string()]), value) + if (typeof exportPaths === "string") { + // special case for "./package.json" + const resolvedPath = path.join(pkgRoot, exportPaths) + expect(await Bun.file(resolvedPath).exists()).toBe(true) + } else { + Object.values(exportPaths).forEach(async (_path: string) => { + const resolvedPath = path.join(pkgRoot, _path) + expect(await Bun.file(resolvedPath).exists()).toBe(true) + }) + } + }) + }) +}) diff --git a/packages/api/build.ts b/packages/api/build.ts new file mode 100644 index 0000000..58e79ab --- /dev/null +++ b/packages/api/build.ts @@ -0,0 +1,20 @@ +import fs from "fs" +import { $ } from "bun" + +// add package version + +if (fs.existsSync("dist")) { + await $`rm -rf dist` +} +fs.mkdirSync("dist") +// await $`pnpm build:rollup` +// await $`cp ../schema/manifest-json-schema.json ./dist/schema.json` +await $`bun ../schema/scripts/print-schema.ts > dist/schema.json` + +// Post Build Verify +const schemaFile = Bun.file("dist/schema.json") +if (!schemaFile.exists()) { + throw new Error("schema.json not found") +} + +await $`bun patch-version.ts` diff --git a/packages/api/deno.lock b/packages/api/deno.lock new file mode 100644 index 0000000..c244861 --- /dev/null +++ b/packages/api/deno.lock @@ -0,0 +1,2361 @@ +{ + "version": "4", + "specifiers": { + "npm:@huakunshen/comlink@^4.4.1": "4.4.1", + "npm:@rollup/plugin-json@^6.1.0": "6.1.0_rollup@4.24.3", + "npm:@rollup/plugin-terser@~0.4.4": "0.4.4_rollup@4.24.3", + "npm:@rollup/plugin-typescript@^11.1.6": "11.1.6_rollup@4.24.3_typescript@5.6.3", + "npm:@tauri-apps/api@^2.0.3": "2.0.3", + "npm:@tauri-apps/cli@^2.0.4": "2.0.4", + "npm:@tauri-apps/plugin-deep-link@2": "2.0.0", + "npm:@tauri-apps/plugin-dialog@^2.0.1": "2.0.1", + "npm:@tauri-apps/plugin-fs@^2.0.1": "2.0.1", + "npm:@tauri-apps/plugin-global-shortcut@2": "2.0.0", + "npm:@tauri-apps/plugin-http@^2.0.1": "2.0.1", + "npm:@tauri-apps/plugin-log@2": "2.0.0", + "npm:@tauri-apps/plugin-notification@2": "2.0.0", + "npm:@tauri-apps/plugin-os@2": "2.0.0", + "npm:@tauri-apps/plugin-process@2.0.0": "2.0.0", + "npm:@tauri-apps/plugin-shell@^2.0.1": "2.0.1", + "npm:@tauri-apps/plugin-store@^2.1.0": "2.1.0", + "npm:@tauri-apps/plugin-updater@2": "2.0.0", + "npm:@tauri-apps/plugin-upload@2": "2.0.0", + "npm:@types/bun@latest": "1.1.12", + "npm:@types/lodash@^4.17.10": "4.17.13", + "npm:@types/madge@^5.0.3": "5.0.3", + "npm:@types/semver@^7.5.8": "7.5.8", + "npm:comlink-stdio@~0.1.7": "0.1.7_typescript@5.6.3", + "npm:comlink@^4.4.1": "4.4.1", + "npm:fs-extra@^11.2.0": "11.2.0", + "npm:lodash@^4.17.21": "4.17.21", + "npm:madge@7": "7.0.0_typescript@5.6.3", + "npm:minimatch@^10.0.1": "10.0.1", + "npm:rimraf@^5.0.7": "5.0.10", + "npm:rollup-plugin-typescript2@0.36": "0.36.0_rollup@4.24.3_typescript@5.6.3", + "npm:rollup-plugin-visualizer@^5.12.0": "5.12.0_rollup@4.24.3", + "npm:rollup@^4.24.0": "4.24.3", + "npm:semver@^7.6.3": "7.6.3", + "npm:svelte-sonner@~0.3.28": "0.3.28_svelte@5.1.5__acorn@8.14.0", + "npm:tauri-api-adapter@0.3.8": "0.3.8_typescript@5.6.3_rollup@4.24.3", + "npm:tauri-plugin-network-api@2.0.3": "2.0.3_typescript@5.6.3", + "npm:tauri-plugin-shellx-api@^2.0.10": "2.0.11", + "npm:tauri-plugin-system-info-api@2.0.6": "2.0.6_typescript@5.6.3", + "npm:tsc-alias@^1.8.10": "1.8.10", + "npm:tsup@^8.1.2": "8.3.5_typescript@5.6.3_esbuild@0.24.0", + "npm:typedoc@~0.26.4": "0.26.10_typescript@5.6.3", + "npm:typescript@5": "5.6.3", + "npm:valibot@0.40": "0.40.0_typescript@5.6.3" + }, + "npm": { + "@ampproject/remapping@2.3.0": { + "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", + "dependencies": [ + "@jridgewell/gen-mapping", + "@jridgewell/trace-mapping" + ] + }, + "@babel/helper-string-parser@7.25.9": { + "integrity": "sha512-4A/SCr/2KLd5jrtOMFzaKjVtAei3+2r/NChoBNoZ3EyP/+GlhoaEGoWOZUmFmoITP7zOJyHIMm+DYRd8o3PvHA==" + }, + "@babel/helper-validator-identifier@7.25.9": { + "integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==" + }, + "@babel/parser@7.26.1": { + "integrity": "sha512-reoQYNiAJreZNsJzyrDNzFQ+IQ5JFiIzAHJg9bn94S3l+4++J7RsIhNMoB+lgP/9tpmiAQqspv+xfdxTSzREOw==", + "dependencies": [ + "@babel/types" + ] + }, + "@babel/types@7.26.0": { + "integrity": "sha512-Z/yiTPj+lDVnF7lWeKCIJzaIkI0vYO87dMpZ4bg4TDrFe4XXLFWL1TbXU27gBP3QccxV9mZICCrnjnYlJjXHOA==", + "dependencies": [ + "@babel/helper-string-parser", + "@babel/helper-validator-identifier" + ] + }, + "@dependents/detective-less@4.1.0": { + "integrity": "sha512-KrkT6qO5NxqNfy68sBl6CTSoJ4SNDIS5iQArkibhlbGU4LaDukZ3q2HIkh8aUKDio6o4itU4xDR7t82Y2eP1Bg==", + "dependencies": [ + "gonzales-pe", + "node-source-walk" + ] + }, + "@esbuild/aix-ppc64@0.24.0": { + "integrity": "sha512-WtKdFM7ls47zkKHFVzMz8opM7LkcsIp9amDUBIAWirg70RM71WRSjdILPsY5Uv1D42ZpUfaPILDlfactHgsRkw==" + }, + "@esbuild/android-arm64@0.24.0": { + "integrity": "sha512-Vsm497xFM7tTIPYK9bNTYJyF/lsP590Qc1WxJdlB6ljCbdZKU9SY8i7+Iin4kyhV/KV5J2rOKsBQbB77Ab7L/w==" + }, + "@esbuild/android-arm@0.24.0": { + "integrity": "sha512-arAtTPo76fJ/ICkXWetLCc9EwEHKaeya4vMrReVlEIUCAUncH7M4bhMQ+M9Vf+FFOZJdTNMXNBrWwW+OXWpSew==" + }, + "@esbuild/android-x64@0.24.0": { + "integrity": "sha512-t8GrvnFkiIY7pa7mMgJd7p8p8qqYIz1NYiAoKc75Zyv73L3DZW++oYMSHPRarcotTKuSs6m3hTOa5CKHaS02TQ==" + }, + "@esbuild/darwin-arm64@0.24.0": { + "integrity": "sha512-CKyDpRbK1hXwv79soeTJNHb5EiG6ct3efd/FTPdzOWdbZZfGhpbcqIpiD0+vwmpu0wTIL97ZRPZu8vUt46nBSw==" + }, + "@esbuild/darwin-x64@0.24.0": { + "integrity": "sha512-rgtz6flkVkh58od4PwTRqxbKH9cOjaXCMZgWD905JOzjFKW+7EiUObfd/Kav+A6Gyud6WZk9w+xu6QLytdi2OA==" + }, + "@esbuild/freebsd-arm64@0.24.0": { + "integrity": "sha512-6Mtdq5nHggwfDNLAHkPlyLBpE5L6hwsuXZX8XNmHno9JuL2+bg2BX5tRkwjyfn6sKbxZTq68suOjgWqCicvPXA==" + }, + "@esbuild/freebsd-x64@0.24.0": { + "integrity": "sha512-D3H+xh3/zphoX8ck4S2RxKR6gHlHDXXzOf6f/9dbFt/NRBDIE33+cVa49Kil4WUjxMGW0ZIYBYtaGCa2+OsQwQ==" + }, + "@esbuild/linux-arm64@0.24.0": { + "integrity": "sha512-TDijPXTOeE3eaMkRYpcy3LarIg13dS9wWHRdwYRnzlwlA370rNdZqbcp0WTyyV/k2zSxfko52+C7jU5F9Tfj1g==" + }, + "@esbuild/linux-arm@0.24.0": { + "integrity": "sha512-gJKIi2IjRo5G6Glxb8d3DzYXlxdEj2NlkixPsqePSZMhLudqPhtZ4BUrpIuTjJYXxvF9njql+vRjB2oaC9XpBw==" + }, + "@esbuild/linux-ia32@0.24.0": { + "integrity": "sha512-K40ip1LAcA0byL05TbCQ4yJ4swvnbzHscRmUilrmP9Am7//0UjPreh4lpYzvThT2Quw66MhjG//20mrufm40mA==" + }, + "@esbuild/linux-loong64@0.24.0": { + "integrity": "sha512-0mswrYP/9ai+CU0BzBfPMZ8RVm3RGAN/lmOMgW4aFUSOQBjA31UP8Mr6DDhWSuMwj7jaWOT0p0WoZ6jeHhrD7g==" + }, + "@esbuild/linux-mips64el@0.24.0": { + "integrity": "sha512-hIKvXm0/3w/5+RDtCJeXqMZGkI2s4oMUGj3/jM0QzhgIASWrGO5/RlzAzm5nNh/awHE0A19h/CvHQe6FaBNrRA==" + }, + "@esbuild/linux-ppc64@0.24.0": { + "integrity": "sha512-HcZh5BNq0aC52UoocJxaKORfFODWXZxtBaaZNuN3PUX3MoDsChsZqopzi5UupRhPHSEHotoiptqikjN/B77mYQ==" + }, + "@esbuild/linux-riscv64@0.24.0": { + "integrity": "sha512-bEh7dMn/h3QxeR2KTy1DUszQjUrIHPZKyO6aN1X4BCnhfYhuQqedHaa5MxSQA/06j3GpiIlFGSsy1c7Gf9padw==" + }, + "@esbuild/linux-s390x@0.24.0": { + "integrity": "sha512-ZcQ6+qRkw1UcZGPyrCiHHkmBaj9SiCD8Oqd556HldP+QlpUIe2Wgn3ehQGVoPOvZvtHm8HPx+bH20c9pvbkX3g==" + }, + "@esbuild/linux-x64@0.24.0": { + "integrity": "sha512-vbutsFqQ+foy3wSSbmjBXXIJ6PL3scghJoM8zCL142cGaZKAdCZHyf+Bpu/MmX9zT9Q0zFBVKb36Ma5Fzfa8xA==" + }, + "@esbuild/netbsd-x64@0.24.0": { + "integrity": "sha512-hjQ0R/ulkO8fCYFsG0FZoH+pWgTTDreqpqY7UnQntnaKv95uP5iW3+dChxnx7C3trQQU40S+OgWhUVwCjVFLvg==" + }, + "@esbuild/openbsd-arm64@0.24.0": { + "integrity": "sha512-MD9uzzkPQbYehwcN583yx3Tu5M8EIoTD+tUgKF982WYL9Pf5rKy9ltgD0eUgs8pvKnmizxjXZyLt0z6DC3rRXg==" + }, + "@esbuild/openbsd-x64@0.24.0": { + "integrity": "sha512-4ir0aY1NGUhIC1hdoCzr1+5b43mw99uNwVzhIq1OY3QcEwPDO3B7WNXBzaKY5Nsf1+N11i1eOfFcq+D/gOS15Q==" + }, + "@esbuild/sunos-x64@0.24.0": { + "integrity": "sha512-jVzdzsbM5xrotH+W5f1s+JtUy1UWgjU0Cf4wMvffTB8m6wP5/kx0KiaLHlbJO+dMgtxKV8RQ/JvtlFcdZ1zCPA==" + }, + "@esbuild/win32-arm64@0.24.0": { + "integrity": "sha512-iKc8GAslzRpBytO2/aN3d2yb2z8XTVfNV0PjGlCxKo5SgWmNXx82I/Q3aG1tFfS+A2igVCY97TJ8tnYwpUWLCA==" + }, + "@esbuild/win32-ia32@0.24.0": { + "integrity": "sha512-vQW36KZolfIudCcTnaTpmLQ24Ha1RjygBo39/aLkM2kmjkWmZGEJ5Gn9l5/7tzXA42QGIoWbICfg6KLLkIw6yw==" + }, + "@esbuild/win32-x64@0.24.0": { + "integrity": "sha512-7IAFPrjSQIJrGsK6flwg7NFmwBoSTyF3rl7If0hNUFQU4ilTsEPL6GuMuU9BfIWVVGuRnuIidkSMC+c0Otu8IA==" + }, + "@huakunshen/comlink@4.4.1": { + "integrity": "sha512-rxWFQ0NEc28z/r6Ixy0ZllNaRqvkqbmuT3oTwzVnwVyDFdIkqoicrwr1Z718ZQbWoVFBdmqSjm93N273ludHMA==" + }, + "@isaacs/cliui@8.0.2": { + "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", + "dependencies": [ + "string-width@5.1.2", + "string-width-cjs@npm:string-width@4.2.3", + "strip-ansi@7.1.0", + "strip-ansi-cjs@npm:strip-ansi@6.0.1", + "wrap-ansi@8.1.0", + "wrap-ansi-cjs@npm:wrap-ansi@7.0.0" + ] + }, + "@jridgewell/gen-mapping@0.3.5": { + "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", + "dependencies": [ + "@jridgewell/set-array", + "@jridgewell/sourcemap-codec", + "@jridgewell/trace-mapping" + ] + }, + "@jridgewell/resolve-uri@3.1.2": { + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==" + }, + "@jridgewell/set-array@1.2.1": { + "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==" + }, + "@jridgewell/source-map@0.3.6": { + "integrity": "sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ==", + "dependencies": [ + "@jridgewell/gen-mapping", + "@jridgewell/trace-mapping" + ] + }, + "@jridgewell/sourcemap-codec@1.5.0": { + "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==" + }, + "@jridgewell/trace-mapping@0.3.25": { + "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", + "dependencies": [ + "@jridgewell/resolve-uri", + "@jridgewell/sourcemap-codec" + ] + }, + "@nodelib/fs.scandir@2.1.5": { + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dependencies": [ + "@nodelib/fs.stat", + "run-parallel" + ] + }, + "@nodelib/fs.stat@2.0.5": { + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==" + }, + "@nodelib/fs.walk@1.2.8": { + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dependencies": [ + "@nodelib/fs.scandir", + "fastq" + ] + }, + "@pkgjs/parseargs@0.11.0": { + "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==" + }, + "@rollup/plugin-alias@5.1.1_rollup@4.24.3": { + "integrity": "sha512-PR9zDb+rOzkRb2VD+EuKB7UC41vU5DIwZ5qqCpk0KJudcWAyi8rvYOhS7+L5aZCspw1stTViLgN5v6FF1p5cgQ==", + "dependencies": [ + "rollup" + ] + }, + "@rollup/plugin-json@6.1.0_rollup@4.24.3": { + "integrity": "sha512-EGI2te5ENk1coGeADSIwZ7G2Q8CJS2sF120T7jLw4xFw9n7wIOXHo+kIYRAoVpJAN+kmqZSoO3Fp4JtoNF4ReA==", + "dependencies": [ + "@rollup/pluginutils@5.1.3_rollup@4.24.3", + "rollup" + ] + }, + "@rollup/plugin-terser@0.4.4_rollup@4.24.3": { + "integrity": "sha512-XHeJC5Bgvs8LfukDwWZp7yeqin6ns8RTl2B9avbejt6tZqsqvVoWI7ZTQrcNsfKEDWBTnTxM8nMDkO2IFFbd0A==", + "dependencies": [ + "rollup", + "serialize-javascript", + "smob", + "terser" + ] + }, + "@rollup/plugin-typescript@11.1.6_rollup@4.24.3_typescript@5.6.3": { + "integrity": "sha512-R92yOmIACgYdJ7dJ97p4K69I8gg6IEHt8M7dUBxN3W6nrO8uUxX5ixl0yU/N3aZTi8WhPuICvOHXQvF6FaykAA==", + "dependencies": [ + "@rollup/pluginutils@5.1.3_rollup@4.24.3", + "resolve", + "rollup", + "typescript" + ] + }, + "@rollup/pluginutils@4.2.1": { + "integrity": "sha512-iKnFXr7NkdZAIHiIWE+BX5ULi/ucVFYWD6TbAV+rZctiRTY2PL6tsIKhoIOaoskiWAkgu+VsbXgUVDNLHf+InQ==", + "dependencies": [ + "estree-walker", + "picomatch@2.3.1" + ] + }, + "@rollup/pluginutils@5.1.3_rollup@4.24.3": { + "integrity": "sha512-Pnsb6f32CD2W3uCaLZIzDmeFyQ2b8UWMFI7xtwUezpcGBDVDW6y9XgAWIlARiGAo6eNF5FK5aQTr0LFyNyqq5A==", + "dependencies": [ + "@types/estree", + "estree-walker", + "picomatch@4.0.2", + "rollup" + ] + }, + "@rollup/rollup-android-arm-eabi@4.24.3": { + "integrity": "sha512-ufb2CH2KfBWPJok95frEZZ82LtDl0A6QKTa8MoM+cWwDZvVGl5/jNb79pIhRvAalUu+7LD91VYR0nwRD799HkQ==" + }, + "@rollup/rollup-android-arm64@4.24.3": { + "integrity": "sha512-iAHpft/eQk9vkWIV5t22V77d90CRofgR2006UiCjHcHJFVI1E0oBkQIAbz+pLtthFw3hWEmVB4ilxGyBf48i2Q==" + }, + "@rollup/rollup-darwin-arm64@4.24.3": { + "integrity": "sha512-QPW2YmkWLlvqmOa2OwrfqLJqkHm7kJCIMq9kOz40Zo9Ipi40kf9ONG5Sz76zszrmIZZ4hgRIkez69YnTHgEz1w==" + }, + "@rollup/rollup-darwin-x64@4.24.3": { + "integrity": "sha512-KO0pN5x3+uZm1ZXeIfDqwcvnQ9UEGN8JX5ufhmgH5Lz4ujjZMAnxQygZAVGemFWn+ZZC0FQopruV4lqmGMshow==" + }, + "@rollup/rollup-freebsd-arm64@4.24.3": { + "integrity": "sha512-CsC+ZdIiZCZbBI+aRlWpYJMSWvVssPuWqrDy/zi9YfnatKKSLFCe6fjna1grHuo/nVaHG+kiglpRhyBQYRTK4A==" + }, + "@rollup/rollup-freebsd-x64@4.24.3": { + "integrity": "sha512-F0nqiLThcfKvRQhZEzMIXOQG4EeX61im61VYL1jo4eBxv4aZRmpin6crnBJQ/nWnCsjH5F6J3W6Stdm0mBNqBg==" + }, + "@rollup/rollup-linux-arm-gnueabihf@4.24.3": { + "integrity": "sha512-KRSFHyE/RdxQ1CSeOIBVIAxStFC/hnBgVcaiCkQaVC+EYDtTe4X7z5tBkFyRoBgUGtB6Xg6t9t2kulnX6wJc6A==" + }, + "@rollup/rollup-linux-arm-musleabihf@4.24.3": { + "integrity": "sha512-h6Q8MT+e05zP5BxEKz0vi0DhthLdrNEnspdLzkoFqGwnmOzakEHSlXfVyA4HJ322QtFy7biUAVFPvIDEDQa6rw==" + }, + "@rollup/rollup-linux-arm64-gnu@4.24.3": { + "integrity": "sha512-fKElSyXhXIJ9pqiYRqisfirIo2Z5pTTve5K438URf08fsypXrEkVmShkSfM8GJ1aUyvjakT+fn2W7Czlpd/0FQ==" + }, + "@rollup/rollup-linux-arm64-musl@4.24.3": { + "integrity": "sha512-YlddZSUk8G0px9/+V9PVilVDC6ydMz7WquxozToozSnfFK6wa6ne1ATUjUvjin09jp34p84milxlY5ikueoenw==" + }, + "@rollup/rollup-linux-powerpc64le-gnu@4.24.3": { + "integrity": "sha512-yNaWw+GAO8JjVx3s3cMeG5Esz1cKVzz8PkTJSfYzE5u7A+NvGmbVFEHP+BikTIyYWuz0+DX9kaA3pH9Sqxp69g==" + }, + "@rollup/rollup-linux-riscv64-gnu@4.24.3": { + "integrity": "sha512-lWKNQfsbpv14ZCtM/HkjCTm4oWTKTfxPmr7iPfp3AHSqyoTz5AgLemYkWLwOBWc+XxBbrU9SCokZP0WlBZM9lA==" + }, + "@rollup/rollup-linux-s390x-gnu@4.24.3": { + "integrity": "sha512-HoojGXTC2CgCcq0Woc/dn12wQUlkNyfH0I1ABK4Ni9YXyFQa86Fkt2Q0nqgLfbhkyfQ6003i3qQk9pLh/SpAYw==" + }, + "@rollup/rollup-linux-x64-gnu@4.24.3": { + "integrity": "sha512-mnEOh4iE4USSccBOtcrjF5nj+5/zm6NcNhbSEfR3Ot0pxBwvEn5QVUXcuOwwPkapDtGZ6pT02xLoPaNv06w7KQ==" + }, + "@rollup/rollup-linux-x64-musl@4.24.3": { + "integrity": "sha512-rMTzawBPimBQkG9NKpNHvquIUTQPzrnPxPbCY1Xt+mFkW7pshvyIS5kYgcf74goxXOQk0CP3EoOC1zcEezKXhw==" + }, + "@rollup/rollup-win32-arm64-msvc@4.24.3": { + "integrity": "sha512-2lg1CE305xNvnH3SyiKwPVsTVLCg4TmNCF1z7PSHX2uZY2VbUpdkgAllVoISD7JO7zu+YynpWNSKAtOrX3AiuA==" + }, + "@rollup/rollup-win32-ia32-msvc@4.24.3": { + "integrity": "sha512-9SjYp1sPyxJsPWuhOCX6F4jUMXGbVVd5obVpoVEi8ClZqo52ViZewA6eFz85y8ezuOA+uJMP5A5zo6Oz4S5rVQ==" + }, + "@rollup/rollup-win32-x64-msvc@4.24.3": { + "integrity": "sha512-HGZgRFFYrMrP3TJlq58nR1xy8zHKId25vhmm5S9jETEfDf6xybPxsavFTJaufe2zgOGYJBskGlj49CwtEuFhWQ==" + }, + "@shikijs/core@1.22.2": { + "integrity": "sha512-bvIQcd8BEeR1yFvOYv6HDiyta2FFVePbzeowf5pPS1avczrPK+cjmaxxh0nx5QzbON7+Sv0sQfQVciO7bN72sg==", + "dependencies": [ + "@shikijs/engine-javascript", + "@shikijs/engine-oniguruma", + "@shikijs/types", + "@shikijs/vscode-textmate", + "@types/hast", + "hast-util-to-html" + ] + }, + "@shikijs/engine-javascript@1.22.2": { + "integrity": "sha512-iOvql09ql6m+3d1vtvP8fLCVCK7BQD1pJFmHIECsujB0V32BJ0Ab6hxk1ewVSMFA58FI0pR2Had9BKZdyQrxTw==", + "dependencies": [ + "@shikijs/types", + "@shikijs/vscode-textmate", + "oniguruma-to-js" + ] + }, + "@shikijs/engine-oniguruma@1.22.2": { + "integrity": "sha512-GIZPAGzQOy56mGvWMoZRPggn0dTlBf1gutV5TdceLCZlFNqWmuc7u+CzD0Gd9vQUTgLbrt0KLzz6FNprqYAxlA==", + "dependencies": [ + "@shikijs/types", + "@shikijs/vscode-textmate" + ] + }, + "@shikijs/types@1.22.2": { + "integrity": "sha512-NCWDa6LGZqTuzjsGfXOBWfjS/fDIbDdmVDug+7ykVe1IKT4c1gakrvlfFYp5NhAXH/lyqLM8wsAPo5wNy73Feg==", + "dependencies": [ + "@shikijs/vscode-textmate", + "@types/hast" + ] + }, + "@shikijs/vscode-textmate@9.3.0": { + "integrity": "sha512-jn7/7ky30idSkd/O5yDBfAnVt+JJpepofP/POZ1iMOxK59cOfqIgg/Dj0eFsjOTMw+4ycJN0uhZH/Eb0bs/EUA==" + }, + "@tauri-apps/api@2.0.1": { + "integrity": "sha512-eoQWT+Tq1qSwQpHV+nw1eNYe5B/nm1PoRjQCRiEOS12I1b+X4PUcREfXVX8dPcBT6GrzWGDtaecY0+1p0Rfqlw==" + }, + "@tauri-apps/api@2.0.3": { + "integrity": "sha512-840qk6n8rbXBXMA5/aAgTYsg5JAubKO0nXw5wf7IzGnUuYKGbB4oFBIZtXOIWy+E0kNTDI3qhq5iqsoMJfwp8g==" + }, + "@tauri-apps/cli-darwin-arm64@2.0.4": { + "integrity": "sha512-siH7rOHobb16rPbc11k64p1mxIpiRCkWmzs2qmL5IX21Gx9K5onI3Tk67Oqpf2uNupbYzItrOttaDT4NHFC7tw==" + }, + "@tauri-apps/cli-darwin-x64@2.0.4": { + "integrity": "sha512-zIccfbCoZMfmUpnk6PFCV0keFyfVj1A9XV3Oiiitj/dkTZ9CQvzjhX3XC0XcK4rsTWegfr2PjSrK06aiPAROAw==" + }, + "@tauri-apps/cli-linux-arm-gnueabihf@2.0.4": { + "integrity": "sha512-fgQqJzefOGWCBNg4yrVA82Rg4s1XQr5K0dc2rCxBhJfa696e8dQ1LDrnWq/AiO5r+uHfVaoQTIUvxxpFicYRSA==" + }, + "@tauri-apps/cli-linux-arm64-gnu@2.0.4": { + "integrity": "sha512-u8wbt5tPA9pI6j+d7jGrfOz9UVCiTp+IYzKNiIqlrDsAjqAUFaNXYHKqOUboeFWEmI4zoCWj6LgpS2OJTQ5FKg==" + }, + "@tauri-apps/cli-linux-arm64-musl@2.0.4": { + "integrity": "sha512-hntF1V8e3V1hlrESm93PsghDhf3lA5pbvFrRfYxU1c+fVD/jRXGVw8BH3O1lW8MWwhEg1YdhKk01oAgsuHLuig==" + }, + "@tauri-apps/cli-linux-x64-gnu@2.0.4": { + "integrity": "sha512-Iq1GGJb+oT1T0ZV8izrgf0cBtlzPCJaWcNueRbf1ZXquMf+FSTyQv+/Lo8rq5T6buOIJOH7cAOTuEWWqiCZteg==" + }, + "@tauri-apps/cli-linux-x64-musl@2.0.4": { + "integrity": "sha512-9NTk6Pf0bSwXqCBdAA+PDYts9HeHebZzIo8mbRzRyUbER6QngG5HZb9Ka36Z1QWtJjdRy6uxSb4zb/9NuTeHfA==" + }, + "@tauri-apps/cli-win32-arm64-msvc@2.0.4": { + "integrity": "sha512-OF2e9oxiBFR8A8wVMOhUx9QGN/I1ZkquWC7gVQBnA56nx9PabJlDT08QBy5UD8USqZFVznnfNr2ehlheQahb3g==" + }, + "@tauri-apps/cli-win32-ia32-msvc@2.0.4": { + "integrity": "sha512-T+hCKB3rFP6q0saHHtR02hm6wr1ZPJ0Mkii3oRTxjPG6BBXoVzHNCYzvdgEGJPTA2sFuAQtJH764NRtNlDMifw==" + }, + "@tauri-apps/cli-win32-x64-msvc@2.0.4": { + "integrity": "sha512-GVaiI3KWRFLomjJmApHqihhYlkJ+7FqhumhVfBO6Z2tWzZjQyVQgTdNp0kYEuW2WoAYEj0dKY6qd4YM33xYcUA==" + }, + "@tauri-apps/cli@2.0.4": { + "integrity": "sha512-Hl9eFXz+O366+6su9PfaSzu2EJdFe1p8K8ghkWmi40dz8VmSE7vsMTaOStD0I71ckSOkh2ICDX7FQTBgjlpjWw==", + "dependencies": [ + "@tauri-apps/cli-darwin-arm64", + "@tauri-apps/cli-darwin-x64", + "@tauri-apps/cli-linux-arm-gnueabihf", + "@tauri-apps/cli-linux-arm64-gnu", + "@tauri-apps/cli-linux-arm64-musl", + "@tauri-apps/cli-linux-x64-gnu", + "@tauri-apps/cli-linux-x64-musl", + "@tauri-apps/cli-win32-arm64-msvc", + "@tauri-apps/cli-win32-ia32-msvc", + "@tauri-apps/cli-win32-x64-msvc" + ] + }, + "@tauri-apps/plugin-deep-link@2.0.0": { + "integrity": "sha512-cDa2k1OrRU5DoKc0IXl1Y8RlFOU107u2phdZfT7FkApsC6TL/VAPs3YOUTT8p9/PZ50EjOKP104HFMqVqnQ0bw==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-dialog@2.0.1": { + "integrity": "sha512-fnUrNr6EfvTqdls/ufusU7h6UbNFzLKvHk/zTuOiBq01R3dTODqwctZlzakdbfSp/7pNwTKvgKTAgl/NAP/Z0Q==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-fs@2.0.1": { + "integrity": "sha512-PkeZG2WAob9Xpmr66aPvj+McDVgFjV2a7YBzYVZjiCvbGeMs6Yk09tlXhCe3EyZdT/pwWMSi8lXUace+hlsjsw==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-global-shortcut@2.0.0": { + "integrity": "sha512-pnB4CUwFVjg4twtBSxoLJ4uLFTYxsvOdC1zIbG581pYzhYatOl6mjB+ijD5SSXgiS/jNoqMcfkOF9PWAisurew==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-http@2.0.1": { + "integrity": "sha512-j6IA3pVBybSCwPpsihpX4z8bs6PluuGtr06ahL/xy4D8HunNBTmRmadJrFOQi0gOAbaig4MkQ15nzNLBLy8R1A==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-log@2.0.0": { + "integrity": "sha512-C+NII9vzswqnOQE8k7oRtnaw0z5TZsMmnirRhXkCKDEhQQH9841Us/PC1WHtGiAaJ8za1A1JB2xXndEq/47X/w==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-notification@2.0.0": { + "integrity": "sha512-6qEDYJS7mgXZWLXA0EFL+DVCJh8sJlzSoyw6B50pxhLPVFjc5Vr5DVzl5W3mUHaYhod5wsC984eQnlCCGqxYDA==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-os@2.0.0": { + "integrity": "sha512-M7hG/nNyQYTJxVG/UhTKhp9mpXriwWzrs9mqDreB8mIgqA3ek5nHLdwRZJWhkKjZrnDT4v9CpA9BhYeplTlAiA==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-process@2.0.0": { + "integrity": "sha512-OYzi0GnkrF4NAnsHZU7U3tjSoP0PbeAlO7T1Z+vJoBUH9sFQ1NSLqWYWQyf8hcb3gVWe7P1JggjiskO+LST1ug==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-shell@2.0.1": { + "integrity": "sha512-akU1b77sw3qHiynrK0s930y8zKmcdrSD60htjH+mFZqv5WaakZA/XxHR3/sF1nNv9Mgmt/Shls37HwnOr00aSw==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-store@2.1.0": { + "integrity": "sha512-GADqrc17opUKYIAKnGHIUgEeTZ2wJGu1ZITKQ1WMuOFdv8fvXRFBAqsqPjE3opgWohbczX6e1NpwmZK1AnuWVw==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-updater@2.0.0": { + "integrity": "sha512-N0cl71g7RPr7zK2Fe5aoIwzw14NcdLcz7XMGFWZVjprsqgDRWoxbnUkknyCQMZthjhGkppCd/wN2MIsUz+eAhQ==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@tauri-apps/plugin-upload@2.0.0": { + "integrity": "sha512-MdBZn4RCVmFDjuC2rbpFGjAZAbePkxrc9T35+SKJJgfXbXYw0YOzvUTsDiU7F3NzDWepO+6EbVN3Upzi9rqt/A==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "@types/bun@1.1.12": { + "integrity": "sha512-UkewJesRDP3+AW30Gc8hvxuIt+vHgYZXmVOKaXV8xnwAnMXTAs3XZDsa/jW+LSdAYhHslokSm72lq63FYYjZqA==", + "dependencies": [ + "bun-types" + ] + }, + "@types/estree@1.0.6": { + "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==" + }, + "@types/hast@3.0.4": { + "integrity": "sha512-WPs+bbQw5aCj+x6laNGWLH3wviHtoCv/P3+otBhbOhJgG8qtpdAMlTCxLtsTWA7LH1Oh/bFCHsBn0TPS5m30EQ==", + "dependencies": [ + "@types/unist" + ] + }, + "@types/lodash@4.17.13": { + "integrity": "sha512-lfx+dftrEZcdBPczf9d0Qv0x+j/rfNCMuC6OcfXmO8gkfeNAY88PgKUbvG56whcN23gc27yenwF6oJZXGFpYxg==" + }, + "@types/madge@5.0.3": { + "integrity": "sha512-NlQJd0qRAoyu+pawTDhLxkW940QT2dqASfwd2g/xEZu2F4Xjwa7TVRSPdbmZwUF1ygvAh0/nepeN7JjwEuOXCA==", + "dependencies": [ + "@types/node@22.5.4" + ] + }, + "@types/mdast@4.0.4": { + "integrity": "sha512-kGaNbPh1k7AFzgpud/gMdvIm5xuECykRR+JnWKQno9TAXVa6WIVCGTPvYGekIDL4uwCZQSYbUxNBSb1aUo79oA==", + "dependencies": [ + "@types/unist" + ] + }, + "@types/node@20.12.14": { + "integrity": "sha512-scnD59RpYD91xngrQQLGkE+6UrHUPzeKZWhhjBSa3HSkwjbQc38+q3RoIVEwxQGRw3M+j5hpNAM+lgV3cVormg==", + "dependencies": [ + "undici-types@5.26.5" + ] + }, + "@types/node@22.5.4": { + "integrity": "sha512-FDuKUJQm/ju9fT/SeX/6+gBzoPzlVCzfzmGkwKvRHQVxi4BntVbyIwf6a4Xn62mrvndLiml6z/UBXIdEVjQLXg==", + "dependencies": [ + "undici-types@6.19.8" + ] + }, + "@types/semver@7.5.8": { + "integrity": "sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==" + }, + "@types/unist@3.0.3": { + "integrity": "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q==" + }, + "@types/ws@8.5.12": { + "integrity": "sha512-3tPRkv1EtkDpzlgyKyI8pGsGZAGPEaXeu0DOj5DI25Ja91bdAYddYHbADRYVrZMRbfW+1l5YwXVDKohDJNQxkQ==", + "dependencies": [ + "@types/node@22.5.4" + ] + }, + "@typescript-eslint/types@5.62.0": { + "integrity": "sha512-87NVngcbVXUahrRTqIK27gD2t5Cu1yuCXxbLcFtCzZGlfyVWWh8mLHkoxzjsB6DDNnvdL+fW8MiwPEJyGJQDgQ==" + }, + "@typescript-eslint/typescript-estree@5.62.0_typescript@5.6.3": { + "integrity": "sha512-CmcQ6uY7b9y694lKdRB8FEel7JbU/40iSAPomu++SjLMntB+2Leay2LO6i8VnJk58MtE9/nQSFIH6jpyRWyYzA==", + "dependencies": [ + "@typescript-eslint/types", + "@typescript-eslint/visitor-keys", + "debug", + "globby", + "is-glob", + "semver@7.6.3", + "tsutils" + ] + }, + "@typescript-eslint/visitor-keys@5.62.0": { + "integrity": "sha512-07ny+LHRzQXepkGg6w0mFY41fVUNBrL2Roj/++7V1txKugfjm/Ci/qSND03r2RhlJhJYMcTn9AhhSSqQp0Ysyw==", + "dependencies": [ + "@typescript-eslint/types", + "eslint-visitor-keys" + ] + }, + "@ungap/structured-clone@1.2.0": { + "integrity": "sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==" + }, + "acorn-typescript@1.4.13_acorn@8.14.0": { + "integrity": "sha512-xsc9Xv0xlVfwp2o7sQ+GCQ1PgbkdcpWdTzrwXxO3xDMTAywVS3oXVOcOHuRjAPkS4P9b+yc/qNF15460v+jp4Q==", + "dependencies": [ + "acorn" + ] + }, + "acorn@8.14.0": { + "integrity": "sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==" + }, + "ansi-regex@5.0.1": { + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==" + }, + "ansi-regex@6.1.0": { + "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==" + }, + "ansi-styles@4.3.0": { + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dependencies": [ + "color-convert" + ] + }, + "ansi-styles@6.2.1": { + "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==" + }, + "any-promise@1.3.0": { + "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==" + }, + "anymatch@3.1.3": { + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dependencies": [ + "normalize-path", + "picomatch@2.3.1" + ] + }, + "app-module-path@2.2.0": { + "integrity": "sha512-gkco+qxENJV+8vFcDiiFhuoSvRXb2a/QPqpSoWhVz829VNJfOTnELbBmPmNKFxf3xdNnw4DWCkzkDaavcX/1YQ==" + }, + "argparse@2.0.1": { + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" + }, + "aria-query@5.3.2": { + "integrity": "sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==" + }, + "array-union@2.1.0": { + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==" + }, + "ast-module-types@5.0.0": { + "integrity": "sha512-JvqziE0Wc0rXQfma0HZC/aY7URXHFuZV84fJRtP8u+lhp0JYCNd5wJzVXP45t0PH0Mej3ynlzvdyITYIu0G4LQ==" + }, + "axobject-query@4.1.0": { + "integrity": "sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==" + }, + "balanced-match@1.0.2": { + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" + }, + "base64-js@1.5.1": { + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" + }, + "binary-extensions@2.3.0": { + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==" + }, + "bl@4.1.0": { + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dependencies": [ + "buffer", + "inherits", + "readable-stream" + ] + }, + "brace-expansion@1.1.11": { + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dependencies": [ + "balanced-match", + "concat-map" + ] + }, + "brace-expansion@2.0.1": { + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dependencies": [ + "balanced-match" + ] + }, + "braces@3.0.3": { + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dependencies": [ + "fill-range" + ] + }, + "buffer-from@1.1.2": { + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" + }, + "buffer@5.7.1": { + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dependencies": [ + "base64-js", + "ieee754" + ] + }, + "bun-types@1.1.32": { + "integrity": "sha512-Lxgux4InO/WRjSAEy3iyDscsnDXR8+3rgNDeZYjPAizFYjUraoNuMl9PuRd9XMgFZgdyQwaUX7/QHmOw5KGFQw==", + "dependencies": [ + "@types/node@20.12.14", + "@types/ws" + ] + }, + "bundle-require@5.0.0_esbuild@0.24.0": { + "integrity": "sha512-GuziW3fSSmopcx4KRymQEJVbZUfqlCqcq7dvs6TYwKRZiegK/2buMxQTPs6MGlNv50wms1699qYO54R8XfRX4w==", + "dependencies": [ + "esbuild", + "load-tsconfig" + ] + }, + "cac@6.7.14": { + "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==" + }, + "ccount@2.0.1": { + "integrity": "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg==" + }, + "chalk@4.1.2": { + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dependencies": [ + "ansi-styles@4.3.0", + "supports-color" + ] + }, + "character-entities-html4@2.1.0": { + "integrity": "sha512-1v7fgQRj6hnSwFpq1Eu0ynr/CDEw0rXo2B61qXrLNdHZmPKgb7fqS1a2JwF0rISo9q77jDI8VMEHoApn8qDoZA==" + }, + "character-entities-legacy@3.0.0": { + "integrity": "sha512-RpPp0asT/6ufRm//AJVwpViZbGM/MkjQFxJccQRHmISF/22NBtsHqAWmL+/pmkPWoIUJdWyeVleTl1wydHATVQ==" + }, + "chokidar@3.6.0": { + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dependencies": [ + "anymatch", + "braces", + "fsevents", + "glob-parent", + "is-binary-path", + "is-glob", + "normalize-path", + "readdirp@3.6.0" + ] + }, + "chokidar@4.0.1": { + "integrity": "sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==", + "dependencies": [ + "readdirp@4.0.2" + ] + }, + "cli-cursor@3.1.0": { + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dependencies": [ + "restore-cursor" + ] + }, + "cli-spinners@2.9.2": { + "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==" + }, + "cliui@8.0.1": { + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dependencies": [ + "string-width@4.2.3", + "strip-ansi@6.0.1", + "wrap-ansi@7.0.0" + ] + }, + "clone@1.0.4": { + "integrity": "sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==" + }, + "color-convert@2.0.1": { + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dependencies": [ + "color-name" + ] + }, + "color-name@1.1.4": { + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" + }, + "comlink-stdio@0.1.7_typescript@5.6.3": { + "integrity": "sha512-LTTyZfEq3KuDKnNSFDsDbJzNh5F6h+MTd13D6K7yRt9zvnP2nxZ5DgyaAaw2H87vQUzJ1T93Se1o3FHk9Ch7cw==", + "dependencies": [ + "typescript" + ] + }, + "comlink@4.4.1": { + "integrity": "sha512-+1dlx0aY5Jo1vHy/tSsIGpSkN4tS9rZSW8FIhG0JH/crs9wwweswIo/POr451r7bZww3hFbPAKnTpimzL/mm4Q==" + }, + "comma-separated-tokens@2.0.3": { + "integrity": "sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg==" + }, + "commander@10.0.1": { + "integrity": "sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==" + }, + "commander@2.20.3": { + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" + }, + "commander@4.1.1": { + "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==" + }, + "commander@7.2.0": { + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==" + }, + "commander@9.5.0": { + "integrity": "sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==" + }, + "commondir@1.0.1": { + "integrity": "sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==" + }, + "concat-map@0.0.1": { + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" + }, + "consola@3.2.3": { + "integrity": "sha512-I5qxpzLv+sJhTVEoLYNcTW+bThDCPsit0vLNKShZx6rLtpilNpmmeTPaeqJb9ZE9dV3DGaeby6Vuhrw38WjeyQ==" + }, + "cross-spawn@7.0.3": { + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dependencies": [ + "path-key", + "shebang-command", + "which" + ] + }, + "debug@4.3.7": { + "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", + "dependencies": [ + "ms" + ] + }, + "deep-extend@0.6.0": { + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==" + }, + "defaults@1.0.4": { + "integrity": "sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==", + "dependencies": [ + "clone" + ] + }, + "define-lazy-prop@2.0.0": { + "integrity": "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==" + }, + "dependency-tree@10.0.9": { + "integrity": "sha512-dwc59FRIsht+HfnTVM0BCjJaEWxdq2YAvEDy4/Hn6CwS3CBWMtFnL3aZGAkQn3XCYxk/YcTDE4jX2Q7bFTwCjA==", + "dependencies": [ + "commander@10.0.1", + "filing-cabinet", + "precinct", + "typescript" + ] + }, + "dequal@2.0.3": { + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==" + }, + "detective-amd@5.0.2": { + "integrity": "sha512-XFd/VEQ76HSpym80zxM68ieB77unNuoMwopU2TFT/ErUk5n4KvUTwW4beafAVUugrjV48l4BmmR0rh2MglBaiA==", + "dependencies": [ + "ast-module-types", + "escodegen", + "get-amd-module-type", + "node-source-walk" + ] + }, + "detective-cjs@5.0.1": { + "integrity": "sha512-6nTvAZtpomyz/2pmEmGX1sXNjaqgMplhQkskq2MLrar0ZAIkHMrDhLXkRiK2mvbu9wSWr0V5/IfiTrZqAQMrmQ==", + "dependencies": [ + "ast-module-types", + "node-source-walk" + ] + }, + "detective-es6@4.0.1": { + "integrity": "sha512-k3Z5tB4LQ8UVHkuMrFOlvb3GgFWdJ9NqAa2YLUU/jTaWJIm+JJnEh4PsMc+6dfT223Y8ACKOaC0qcj7diIhBKw==", + "dependencies": [ + "node-source-walk" + ] + }, + "detective-postcss@6.1.3_postcss@8.4.47": { + "integrity": "sha512-7BRVvE5pPEvk2ukUWNQ+H2XOq43xENWbH0LcdCE14mwgTBEAMoAx+Fc1rdp76SmyZ4Sp48HlV7VedUnP6GA1Tw==", + "dependencies": [ + "is-url", + "postcss", + "postcss-values-parser" + ] + }, + "detective-sass@5.0.3": { + "integrity": "sha512-YsYT2WuA8YIafp2RVF5CEfGhhyIVdPzlwQgxSjK+TUm3JoHP+Tcorbk3SfG0cNZ7D7+cYWa0ZBcvOaR0O8+LlA==", + "dependencies": [ + "gonzales-pe", + "node-source-walk" + ] + }, + "detective-scss@4.0.3": { + "integrity": "sha512-VYI6cHcD0fLokwqqPFFtDQhhSnlFWvU614J42eY6G0s8c+MBhi9QAWycLwIOGxlmD8I/XvGSOUV1kIDhJ70ZPg==", + "dependencies": [ + "gonzales-pe", + "node-source-walk" + ] + }, + "detective-stylus@4.0.0": { + "integrity": "sha512-TfPotjhszKLgFBzBhTOxNHDsutIxx9GTWjrL5Wh7Qx/ydxKhwUrlSFeLIn+ZaHPF+h0siVBkAQSuy6CADyTxgQ==" + }, + "detective-typescript@11.2.0_typescript@5.6.3": { + "integrity": "sha512-ARFxjzizOhPqs1fYC/2NMC3N4jrQ6HvVflnXBTRqNEqJuXwyKLRr9CrJwkRcV/SnZt1sNXgsF6FPm0x57Tq0rw==", + "dependencies": [ + "@typescript-eslint/typescript-estree", + "ast-module-types", + "node-source-walk", + "typescript" + ] + }, + "devlop@1.1.0": { + "integrity": "sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA==", + "dependencies": [ + "dequal" + ] + }, + "dir-glob@3.0.1": { + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dependencies": [ + "path-type" + ] + }, + "eastasianwidth@0.2.0": { + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==" + }, + "emoji-regex@8.0.0": { + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "emoji-regex@9.2.2": { + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==" + }, + "enhanced-resolve@5.17.1": { + "integrity": "sha512-LMHl3dXhTcfv8gM4kEzIUeTQ+7fpdA0l2tUf34BddXPkz2A5xJ5L/Pchd5BL6rdccM9QGvu0sWZzK1Z1t4wwyg==", + "dependencies": [ + "graceful-fs", + "tapable" + ] + }, + "entities@4.5.0": { + "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==" + }, + "esbuild@0.24.0": { + "integrity": "sha512-FuLPevChGDshgSicjisSooU0cemp/sGXR841D5LHMB7mTVOmsEHcAxaH3irL53+8YDIeVNQEySh4DaYU/iuPqQ==", + "dependencies": [ + "@esbuild/aix-ppc64", + "@esbuild/android-arm", + "@esbuild/android-arm64", + "@esbuild/android-x64", + "@esbuild/darwin-arm64", + "@esbuild/darwin-x64", + "@esbuild/freebsd-arm64", + "@esbuild/freebsd-x64", + "@esbuild/linux-arm", + "@esbuild/linux-arm64", + "@esbuild/linux-ia32", + "@esbuild/linux-loong64", + "@esbuild/linux-mips64el", + "@esbuild/linux-ppc64", + "@esbuild/linux-riscv64", + "@esbuild/linux-s390x", + "@esbuild/linux-x64", + "@esbuild/netbsd-x64", + "@esbuild/openbsd-arm64", + "@esbuild/openbsd-x64", + "@esbuild/sunos-x64", + "@esbuild/win32-arm64", + "@esbuild/win32-ia32", + "@esbuild/win32-x64" + ] + }, + "escalade@3.2.0": { + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==" + }, + "escodegen@2.1.0": { + "integrity": "sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w==", + "dependencies": [ + "esprima", + "estraverse", + "esutils", + "source-map@0.6.1" + ] + }, + "eslint-visitor-keys@3.4.3": { + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==" + }, + "esm-env@1.0.0": { + "integrity": "sha512-Cf6VksWPsTuW01vU9Mk/3vRue91Zevka5SjyNf3nEpokFRuqt/KjUQoGAwq9qMmhpLTHmXzSIrFRw8zxWzmFBA==" + }, + "esprima@4.0.1": { + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==" + }, + "esrap@1.2.2": { + "integrity": "sha512-F2pSJklxx1BlQIQgooczXCPHmcWpn6EsP5oo73LQfonG9fIlIENQ8vMmfGXeojP9MrkzUNAfyU5vdFlR9shHAw==", + "dependencies": [ + "@jridgewell/sourcemap-codec", + "@types/estree" + ] + }, + "estraverse@5.3.0": { + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==" + }, + "estree-walker@2.0.2": { + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==" + }, + "esutils@2.0.3": { + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==" + }, + "fast-glob@3.3.2": { + "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", + "dependencies": [ + "@nodelib/fs.stat", + "@nodelib/fs.walk", + "glob-parent", + "merge2", + "micromatch" + ] + }, + "fastq@1.17.1": { + "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "dependencies": [ + "reusify" + ] + }, + "fdir@6.4.2_picomatch@4.0.2": { + "integrity": "sha512-KnhMXsKSPZlAhp7+IjUkRZKPb4fUyccpDrdFXbi4QL1qkmFh9kVY09Yox+n4MaOb3lHZ1Tv829C3oaaXoMYPDQ==", + "dependencies": [ + "picomatch@4.0.2" + ] + }, + "filing-cabinet@4.2.0": { + "integrity": "sha512-YZ21ryzRcyqxpyKggdYSoXx//d3sCJzM3lsYoaeg/FyXdADGJrUl+BW1KIglaVLJN5BBcMtWylkygY8zBp2MrQ==", + "dependencies": [ + "app-module-path", + "commander@10.0.1", + "enhanced-resolve", + "is-relative-path", + "module-definition", + "module-lookup-amd", + "resolve", + "resolve-dependency-path", + "sass-lookup", + "stylus-lookup", + "tsconfig-paths", + "typescript" + ] + }, + "fill-range@7.1.1": { + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dependencies": [ + "to-regex-range" + ] + }, + "find-cache-dir@3.3.2": { + "integrity": "sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==", + "dependencies": [ + "commondir", + "make-dir", + "pkg-dir" + ] + }, + "find-up@4.1.0": { + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dependencies": [ + "locate-path", + "path-exists" + ] + }, + "foreground-child@3.3.0": { + "integrity": "sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==", + "dependencies": [ + "cross-spawn", + "signal-exit@4.1.0" + ] + }, + "fs-extra@10.1.0": { + "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==", + "dependencies": [ + "graceful-fs", + "jsonfile", + "universalify" + ] + }, + "fs-extra@11.2.0": { + "integrity": "sha512-PmDi3uwK5nFuXh7XDTlVnS17xJS7vW36is2+w3xcv8SVxiB4NyATf4ctkVY5bkSjX0Y4nbvZCq1/EjtEyr9ktw==", + "dependencies": [ + "graceful-fs", + "jsonfile", + "universalify" + ] + }, + "fs.realpath@1.0.0": { + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==" + }, + "fsevents@2.3.3": { + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==" + }, + "function-bind@1.1.2": { + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==" + }, + "get-amd-module-type@5.0.1": { + "integrity": "sha512-jb65zDeHyDjFR1loOVk0HQGM5WNwoGB8aLWy3LKCieMKol0/ProHkhO2X1JxojuN10vbz1qNn09MJ7tNp7qMzw==", + "dependencies": [ + "ast-module-types", + "node-source-walk" + ] + }, + "get-caller-file@2.0.5": { + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==" + }, + "get-own-enumerable-property-symbols@3.0.2": { + "integrity": "sha512-I0UBV/XOz1XkIJHEUDMZAbzCThU/H8DxmSfmdGcKPnVhu2VfFqr34jr9777IyaTYvxjedWhqVIilEDsCdP5G6g==" + }, + "glob-parent@5.1.2": { + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dependencies": [ + "is-glob" + ] + }, + "glob@10.4.5": { + "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==", + "dependencies": [ + "foreground-child", + "jackspeak", + "minimatch@9.0.5", + "minipass", + "package-json-from-dist", + "path-scurry" + ] + }, + "glob@7.2.3": { + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dependencies": [ + "fs.realpath", + "inflight", + "inherits", + "minimatch@3.1.2", + "once", + "path-is-absolute" + ] + }, + "globby@11.1.0": { + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dependencies": [ + "array-union", + "dir-glob", + "fast-glob", + "ignore", + "merge2", + "slash" + ] + }, + "gonzales-pe@4.3.0": { + "integrity": "sha512-otgSPpUmdWJ43VXyiNgEYE4luzHCL2pz4wQ0OnDluC6Eg4Ko3Vexy/SrSynglw/eR+OhkzmqFCZa/OFa/RgAOQ==", + "dependencies": [ + "minimist" + ] + }, + "graceful-fs@4.2.11": { + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==" + }, + "has-flag@4.0.0": { + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" + }, + "hasown@2.0.2": { + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dependencies": [ + "function-bind" + ] + }, + "hast-util-to-html@9.0.3": { + "integrity": "sha512-M17uBDzMJ9RPCqLMO92gNNUDuBSq10a25SDBI08iCCxmorf4Yy6sYHK57n9WAbRAAaU+DuR4W6GN9K4DFZesYg==", + "dependencies": [ + "@types/hast", + "@types/unist", + "ccount", + "comma-separated-tokens", + "hast-util-whitespace", + "html-void-elements", + "mdast-util-to-hast", + "property-information", + "space-separated-tokens", + "stringify-entities", + "zwitch" + ] + }, + "hast-util-whitespace@3.0.0": { + "integrity": "sha512-88JUN06ipLwsnv+dVn+OIYOvAuvBMy/Qoi6O7mQHxdPXpjy+Cd6xRkWwux7DKO+4sYILtLBRIKgsdpS2gQc7qw==", + "dependencies": [ + "@types/hast" + ] + }, + "html-void-elements@3.0.0": { + "integrity": "sha512-bEqo66MRXsUGxWHV5IP0PUiAWwoEjba4VCzg0LjFJBpchPaTfyfCKTG6bc5F8ucKec3q5y6qOdGyYTSBEvhCrg==" + }, + "ieee754@1.2.1": { + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" + }, + "ignore@5.3.2": { + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==" + }, + "inflight@1.0.6": { + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dependencies": [ + "once", + "wrappy" + ] + }, + "inherits@2.0.4": { + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "ini@1.3.8": { + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==" + }, + "interpret@1.4.0": { + "integrity": "sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==" + }, + "is-binary-path@2.1.0": { + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dependencies": [ + "binary-extensions" + ] + }, + "is-core-module@2.15.1": { + "integrity": "sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==", + "dependencies": [ + "hasown" + ] + }, + "is-docker@2.2.1": { + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==" + }, + "is-extglob@2.1.1": { + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==" + }, + "is-fullwidth-code-point@3.0.0": { + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" + }, + "is-glob@4.0.3": { + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dependencies": [ + "is-extglob" + ] + }, + "is-interactive@1.0.0": { + "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==" + }, + "is-number@7.0.0": { + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" + }, + "is-obj@1.0.1": { + "integrity": "sha512-l4RyHgRqGN4Y3+9JHVrNqO+tN0rV5My76uW5/nuO4K1b6vw5G8d/cmFjP9tRfEsdhZNt0IFdZuK/c2Vr4Nb+Qg==" + }, + "is-reference@3.0.2": { + "integrity": "sha512-v3rht/LgVcsdZa3O2Nqs+NMowLOxeOm7Ay9+/ARQ2F+qEoANRcqrjAZKGN0v8ymUetZGgkp26LTnGT7H0Qo9Pg==", + "dependencies": [ + "@types/estree" + ] + }, + "is-regexp@1.0.0": { + "integrity": "sha512-7zjFAPO4/gwyQAAgRRmqeEeyIICSdmCqa3tsVHMdBzaXXRiqopZL4Cyghg/XulGWrtABTpbnYYzzIRffLkP4oA==" + }, + "is-relative-path@1.0.2": { + "integrity": "sha512-i1h+y50g+0hRbBD+dbnInl3JlJ702aar58snAeX+MxBAPvzXGej7sYoPMhlnykabt0ZzCJNBEyzMlekuQZN7fA==" + }, + "is-unicode-supported@0.1.0": { + "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==" + }, + "is-url-superb@4.0.0": { + "integrity": "sha512-GI+WjezhPPcbM+tqE9LnmsY5qqjwHzTvjJ36wxYX5ujNXefSUJ/T17r5bqDV8yLhcgB59KTPNOc9O9cmHTPWsA==" + }, + "is-url@1.2.4": { + "integrity": "sha512-ITvGim8FhRiYe4IQ5uHSkj7pVaPDrCTkNd3yq3cV7iZAcJdHTUMPMEHcqSOy9xZ9qFenQCvi+2wjH9a1nXqHww==" + }, + "is-wsl@2.2.0": { + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "dependencies": [ + "is-docker" + ] + }, + "isexe@2.0.0": { + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==" + }, + "jackspeak@3.4.3": { + "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==", + "dependencies": [ + "@isaacs/cliui", + "@pkgjs/parseargs" + ] + }, + "joycon@3.1.1": { + "integrity": "sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==" + }, + "json5@2.2.3": { + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==" + }, + "jsonfile@6.1.0": { + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dependencies": [ + "graceful-fs", + "universalify" + ] + }, + "lilconfig@3.1.2": { + "integrity": "sha512-eop+wDAvpItUys0FWkHIKeC9ybYrTGbU41U5K7+bttZZeohvnY7M9dZ5kB21GNWiFT2q1OoPTvncPCgSOVO5ow==" + }, + "lines-and-columns@1.2.4": { + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==" + }, + "linkify-it@5.0.0": { + "integrity": "sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ==", + "dependencies": [ + "uc.micro" + ] + }, + "load-tsconfig@0.2.5": { + "integrity": "sha512-IXO6OCs9yg8tMKzfPZ1YmheJbZCiEsnBdcB03l0OcfK9prKnJb96siuHCr5Fl37/yo9DnKU+TLpxzTUspw9shg==" + }, + "locate-character@3.0.0": { + "integrity": "sha512-SW13ws7BjaeJ6p7Q6CO2nchbYEc3X3J6WrmTTDto7yMPqVSZTUyY5Tjbid+Ab8gLnATtygYtiDIJGQRRn2ZOiA==" + }, + "locate-path@5.0.0": { + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dependencies": [ + "p-locate" + ] + }, + "lodash.sortby@4.7.0": { + "integrity": "sha512-HDWXG8isMntAyRF5vZ7xKuEvOhT4AhlRt/3czTSjvGUxjYCBVRQY48ViDHyfYz9VIoBkW4TMGQNapx+l3RUwdA==" + }, + "lodash@4.17.21": { + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, + "log-symbols@4.1.0": { + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", + "dependencies": [ + "chalk", + "is-unicode-supported" + ] + }, + "lru-cache@10.4.3": { + "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==" + }, + "lunr@2.3.9": { + "integrity": "sha512-zTU3DaZaF3Rt9rhN3uBMGQD3dD2/vFQqnvZCDv4dl5iOzq2IZQqTxu90r4E5J+nP70J3ilqVCrbho2eWaeW8Ow==" + }, + "madge@7.0.0_typescript@5.6.3": { + "integrity": "sha512-x9eHkBWoCJ2B8yGesWf8LRucarkbH5P3lazqgvmxe4xn5U2Meyfu906iG9mBB1RnY/f4D+gtELWdiz1k6+jAZA==", + "dependencies": [ + "chalk", + "commander@7.2.0", + "commondir", + "debug", + "dependency-tree", + "ora", + "pluralize", + "precinct", + "pretty-ms", + "rc", + "stream-to-array", + "ts-graphviz", + "typescript", + "walkdir" + ] + }, + "magic-string@0.30.12": { + "integrity": "sha512-Ea8I3sQMVXr8JhN4z+H/d8zwo+tYDgHE9+5G4Wnrwhs0gaK9fXTKx0Tw5Xwsd/bCPTTZNRAdpyzvoeORe9LYpw==", + "dependencies": [ + "@jridgewell/sourcemap-codec" + ] + }, + "make-dir@3.1.0": { + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dependencies": [ + "semver@6.3.1" + ] + }, + "markdown-it@14.1.0": { + "integrity": "sha512-a54IwgWPaeBCAAsv13YgmALOF1elABB08FxO9i+r4VFk5Vl4pKokRPeX8u5TCgSsPi6ec1otfLjdOpVcgbpshg==", + "dependencies": [ + "argparse", + "entities", + "linkify-it", + "mdurl", + "punycode.js", + "uc.micro" + ] + }, + "mdast-util-to-hast@13.2.0": { + "integrity": "sha512-QGYKEuUsYT9ykKBCMOEDLsU5JRObWQusAolFMeko/tYPufNkRffBAQjIE+99jbA87xv6FgmjLtwjh9wBWajwAA==", + "dependencies": [ + "@types/hast", + "@types/mdast", + "@ungap/structured-clone", + "devlop", + "micromark-util-sanitize-uri", + "trim-lines", + "unist-util-position", + "unist-util-visit", + "vfile" + ] + }, + "mdurl@2.0.0": { + "integrity": "sha512-Lf+9+2r+Tdp5wXDXC4PcIBjTDtq4UKjCPMQhKIuzpJNW0b96kVqSwW0bT7FhRSfmAiFYgP+SCRvdrDozfh0U5w==" + }, + "merge2@1.4.1": { + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==" + }, + "micromark-util-character@2.1.0": { + "integrity": "sha512-KvOVV+X1yLBfs9dCBSopq/+G1PcgT3lAK07mC4BzXi5E7ahzMAF8oIupDDJ6mievI6F+lAATkbQQlQixJfT3aQ==", + "dependencies": [ + "micromark-util-symbol", + "micromark-util-types" + ] + }, + "micromark-util-encode@2.0.0": { + "integrity": "sha512-pS+ROfCXAGLWCOc8egcBvT0kf27GoWMqtdarNfDcjb6YLuV5cM3ioG45Ys2qOVqeqSbjaKg72vU+Wby3eddPsA==" + }, + "micromark-util-sanitize-uri@2.0.0": { + "integrity": "sha512-WhYv5UEcZrbAtlsnPuChHUAsu/iBPOVaEVsntLBIdpibO0ddy8OzavZz3iL2xVvBZOpolujSliP65Kq0/7KIYw==", + "dependencies": [ + "micromark-util-character", + "micromark-util-encode", + "micromark-util-symbol" + ] + }, + "micromark-util-symbol@2.0.0": { + "integrity": "sha512-8JZt9ElZ5kyTnO94muPxIGS8oyElRJaiJO8EzV6ZSyGQ1Is8xwl4Q45qU5UOg+bGH4AikWziz0iN4sFLWs8PGw==" + }, + "micromark-util-types@2.0.0": { + "integrity": "sha512-oNh6S2WMHWRZrmutsRmDDfkzKtxF+bc2VxLC9dvtrDIRFln627VsFP6fLMgTryGDljgLPjkrzQSDcPrjPyDJ5w==" + }, + "micromatch@4.0.8": { + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dependencies": [ + "braces", + "picomatch@2.3.1" + ] + }, + "mimic-fn@2.1.0": { + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==" + }, + "minimatch@10.0.1": { + "integrity": "sha512-ethXTt3SGGR+95gudmqJ1eNhRO7eGEGIgYA9vnPatK4/etz2MEVDno5GMCibdMTuBMyElzIlgxMna3K94XDIDQ==", + "dependencies": [ + "brace-expansion@2.0.1" + ] + }, + "minimatch@3.1.2": { + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dependencies": [ + "brace-expansion@1.1.11" + ] + }, + "minimatch@9.0.5": { + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "dependencies": [ + "brace-expansion@2.0.1" + ] + }, + "minimist@1.2.8": { + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==" + }, + "minipass@7.1.2": { + "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==" + }, + "module-definition@5.0.1": { + "integrity": "sha512-kvw3B4G19IXk+BOXnYq/D/VeO9qfHaapMeuS7w7sNUqmGaA6hywdFHMi+VWeR9wUScXM7XjoryTffCZ5B0/8IA==", + "dependencies": [ + "ast-module-types", + "node-source-walk" + ] + }, + "module-lookup-amd@8.0.5": { + "integrity": "sha512-vc3rYLjDo5Frjox8NZpiyLXsNWJ5BWshztc/5KSOMzpg9k5cHH652YsJ7VKKmtM4SvaxuE9RkrYGhiSjH3Ehow==", + "dependencies": [ + "commander@10.0.1", + "glob@7.2.3", + "requirejs", + "requirejs-config-file" + ] + }, + "ms@2.1.3": { + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" + }, + "mylas@2.1.13": { + "integrity": "sha512-+MrqnJRtxdF+xngFfUUkIMQrUUL0KsxbADUkn23Z/4ibGg192Q+z+CQyiYwvWTsYjJygmMR8+w3ZDa98Zh6ESg==" + }, + "mz@2.7.0": { + "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==", + "dependencies": [ + "any-promise", + "object-assign", + "thenify-all" + ] + }, + "nanoid@3.3.7": { + "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==" + }, + "node-source-walk@6.0.2": { + "integrity": "sha512-jn9vOIK/nfqoFCcpK89/VCVaLg1IHE6UVfDOzvqmANaJ/rWCTEdH8RZ1V278nv2jr36BJdyQXIAavBLXpzdlag==", + "dependencies": [ + "@babel/parser" + ] + }, + "normalize-path@3.0.0": { + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" + }, + "object-assign@4.1.1": { + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==" + }, + "once@1.4.0": { + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dependencies": [ + "wrappy" + ] + }, + "onetime@5.1.2": { + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dependencies": [ + "mimic-fn" + ] + }, + "oniguruma-to-js@0.4.3": { + "integrity": "sha512-X0jWUcAlxORhOqqBREgPMgnshB7ZGYszBNspP+tS9hPD3l13CdaXcHbgImoHUHlrvGx/7AvFEkTRhAGYh+jzjQ==", + "dependencies": [ + "regex" + ] + }, + "open@8.4.2": { + "integrity": "sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==", + "dependencies": [ + "define-lazy-prop", + "is-docker", + "is-wsl" + ] + }, + "ora@5.4.1": { + "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==", + "dependencies": [ + "bl", + "chalk", + "cli-cursor", + "cli-spinners", + "is-interactive", + "is-unicode-supported", + "log-symbols", + "strip-ansi@6.0.1", + "wcwidth" + ] + }, + "p-limit@2.3.0": { + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dependencies": [ + "p-try" + ] + }, + "p-locate@4.1.0": { + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dependencies": [ + "p-limit" + ] + }, + "p-try@2.2.0": { + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==" + }, + "package-json-from-dist@1.0.1": { + "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==" + }, + "parse-ms@2.1.0": { + "integrity": "sha512-kHt7kzLoS9VBZfUsiKjv43mr91ea+U05EyKkEtqp7vNbHxmaVuEqN7XxeEVnGrMtYOAxGrDElSi96K7EgO1zCA==" + }, + "path-exists@4.0.0": { + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==" + }, + "path-is-absolute@1.0.1": { + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==" + }, + "path-key@3.1.1": { + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==" + }, + "path-parse@1.0.7": { + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==" + }, + "path-scurry@1.11.1": { + "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", + "dependencies": [ + "lru-cache", + "minipass" + ] + }, + "path-type@4.0.0": { + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==" + }, + "picocolors@1.1.1": { + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==" + }, + "picomatch@2.3.1": { + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==" + }, + "picomatch@4.0.2": { + "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==" + }, + "pirates@4.0.6": { + "integrity": "sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==" + }, + "pkg-dir@4.2.0": { + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dependencies": [ + "find-up" + ] + }, + "plimit-lit@1.6.1": { + "integrity": "sha512-B7+VDyb8Tl6oMJT9oSO2CW8XC/T4UcJGrwOVoNGwOQsQYhlpfajmrMj5xeejqaASq3V/EqThyOeATEOMuSEXiA==", + "dependencies": [ + "queue-lit" + ] + }, + "pluralize@8.0.0": { + "integrity": "sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==" + }, + "postcss-load-config@6.0.1": { + "integrity": "sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==", + "dependencies": [ + "lilconfig" + ] + }, + "postcss-values-parser@6.0.2_postcss@8.4.47": { + "integrity": "sha512-YLJpK0N1brcNJrs9WatuJFtHaV9q5aAOj+S4DI5S7jgHlRfm0PIbDCAFRYMQD5SHq7Fy6xsDhyutgS0QOAs0qw==", + "dependencies": [ + "color-name", + "is-url-superb", + "postcss", + "quote-unquote" + ] + }, + "postcss@8.4.47": { + "integrity": "sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==", + "dependencies": [ + "nanoid", + "picocolors", + "source-map-js" + ] + }, + "precinct@11.0.5": { + "integrity": "sha512-oHSWLC8cL/0znFhvln26D14KfCQFFn4KOLSw6hmLhd+LQ2SKt9Ljm89but76Pc7flM9Ty1TnXyrA2u16MfRV3w==", + "dependencies": [ + "@dependents/detective-less", + "commander@10.0.1", + "detective-amd", + "detective-cjs", + "detective-es6", + "detective-postcss", + "detective-sass", + "detective-scss", + "detective-stylus", + "detective-typescript", + "module-definition", + "node-source-walk" + ] + }, + "pretty-ms@7.0.1": { + "integrity": "sha512-973driJZvxiGOQ5ONsFhOF/DtzPMOMtgC11kCpUrPGMTgqp2q/1gwzCquocrN33is0VZ5GFHXZYMM9l6h67v2Q==", + "dependencies": [ + "parse-ms" + ] + }, + "property-information@6.5.0": { + "integrity": "sha512-PgTgs/BlvHxOu8QuEN7wi5A0OmXaBcHpmCSTehcs6Uuu9IkDIEo13Hy7n898RHfrQ49vKCoGeWZSaAK01nwVig==" + }, + "punycode.js@2.3.1": { + "integrity": "sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA==" + }, + "punycode@2.3.1": { + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==" + }, + "queue-lit@1.5.2": { + "integrity": "sha512-tLc36IOPeMAubu8BkW8YDBV+WyIgKlYU7zUNs0J5Vk9skSZ4JfGlPOqplP0aHdfv7HL0B2Pg6nwiq60Qc6M2Hw==" + }, + "queue-microtask@1.2.3": { + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==" + }, + "quote-unquote@1.0.0": { + "integrity": "sha512-twwRO/ilhlG/FIgYeKGFqyHhoEhqgnKVkcmqMKi2r524gz3ZbDTcyFt38E9xjJI2vT+KbRNHVbnJ/e0I25Azwg==" + }, + "randombytes@2.1.0": { + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dependencies": [ + "safe-buffer" + ] + }, + "rc@1.2.8": { + "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", + "dependencies": [ + "deep-extend", + "ini", + "minimist", + "strip-json-comments" + ] + }, + "readable-stream@3.6.2": { + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dependencies": [ + "inherits", + "string_decoder", + "util-deprecate" + ] + }, + "readdirp@3.6.0": { + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dependencies": [ + "picomatch@2.3.1" + ] + }, + "readdirp@4.0.2": { + "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==" + }, + "rechoir@0.6.2": { + "integrity": "sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==", + "dependencies": [ + "resolve" + ] + }, + "regex@4.3.3": { + "integrity": "sha512-r/AadFO7owAq1QJVeZ/nq9jNS1vyZt+6t1p/E59B56Rn2GCya+gr1KSyOzNL/er+r+B7phv5jG2xU2Nz1YkmJg==" + }, + "require-directory@2.1.1": { + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==" + }, + "requirejs-config-file@4.0.0": { + "integrity": "sha512-jnIre8cbWOyvr8a5F2KuqBnY+SDA4NXr/hzEZJG79Mxm2WiFQz2dzhC8ibtPJS7zkmBEl1mxSwp5HhC1W4qpxw==", + "dependencies": [ + "esprima", + "stringify-object" + ] + }, + "requirejs@2.3.7": { + "integrity": "sha512-DouTG8T1WanGok6Qjg2SXuCMzszOo0eHeH9hDZ5Y4x8Je+9JB38HdTLT4/VA8OaUhBa0JPVHJ0pyBkM1z+pDsw==" + }, + "resolve-dependency-path@3.0.2": { + "integrity": "sha512-Tz7zfjhLfsvR39ADOSk9us4421J/1ztVBo4rWUkF38hgHK5m0OCZ3NxFVpqHRkjctnwVa15igEUHFJp8MCS7vA==" + }, + "resolve-from@5.0.0": { + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==" + }, + "resolve@1.22.8": { + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", + "dependencies": [ + "is-core-module", + "path-parse", + "supports-preserve-symlinks-flag" + ] + }, + "restore-cursor@3.1.0": { + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dependencies": [ + "onetime", + "signal-exit@3.0.7" + ] + }, + "reusify@1.0.4": { + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==" + }, + "rimraf@5.0.10": { + "integrity": "sha512-l0OE8wL34P4nJH/H2ffoaniAokM2qSmrtXHmlpvYr5AVVX8msAyW0l8NVJFDxlSK4u3Uh/f41cQheDVdnYijwQ==", + "dependencies": [ + "glob@10.4.5" + ] + }, + "rollup-plugin-typescript2@0.36.0_rollup@4.24.3_typescript@5.6.3": { + "integrity": "sha512-NB2CSQDxSe9+Oe2ahZbf+B4bh7pHwjV5L+RSYpCu7Q5ROuN94F9b6ioWwKfz3ueL3KTtmX4o2MUH2cgHDIEUsw==", + "dependencies": [ + "@rollup/pluginutils@4.2.1", + "find-cache-dir", + "fs-extra@10.1.0", + "rollup", + "semver@7.6.3", + "tslib@2.8.0", + "typescript" + ] + }, + "rollup-plugin-visualizer@5.12.0_rollup@4.24.3": { + "integrity": "sha512-8/NU9jXcHRs7Nnj07PF2o4gjxmm9lXIrZ8r175bT9dK8qoLlvKTwRMArRCMgpMGlq8CTLugRvEmyMeMXIU2pNQ==", + "dependencies": [ + "open", + "picomatch@2.3.1", + "rollup", + "source-map@0.7.4", + "yargs" + ] + }, + "rollup@4.24.3": { + "integrity": "sha512-HBW896xR5HGmoksbi3JBDtmVzWiPAYqp7wip50hjQ67JbDz61nyoMPdqu1DvVW9asYb2M65Z20ZHsyJCMqMyDg==", + "dependencies": [ + "@rollup/rollup-android-arm-eabi", + "@rollup/rollup-android-arm64", + "@rollup/rollup-darwin-arm64", + "@rollup/rollup-darwin-x64", + "@rollup/rollup-freebsd-arm64", + "@rollup/rollup-freebsd-x64", + "@rollup/rollup-linux-arm-gnueabihf", + "@rollup/rollup-linux-arm-musleabihf", + "@rollup/rollup-linux-arm64-gnu", + "@rollup/rollup-linux-arm64-musl", + "@rollup/rollup-linux-powerpc64le-gnu", + "@rollup/rollup-linux-riscv64-gnu", + "@rollup/rollup-linux-s390x-gnu", + "@rollup/rollup-linux-x64-gnu", + "@rollup/rollup-linux-x64-musl", + "@rollup/rollup-win32-arm64-msvc", + "@rollup/rollup-win32-ia32-msvc", + "@rollup/rollup-win32-x64-msvc", + "@types/estree", + "fsevents" + ] + }, + "run-parallel@1.2.0": { + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dependencies": [ + "queue-microtask" + ] + }, + "safe-buffer@5.2.1": { + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" + }, + "sass-lookup@5.0.1": { + "integrity": "sha512-t0X5PaizPc2H4+rCwszAqHZRtr4bugo4pgiCvrBFvIX0XFxnr29g77LJcpyj9A0DcKf7gXMLcgvRjsonYI6x4g==", + "dependencies": [ + "commander@10.0.1" + ] + }, + "semver@6.3.1": { + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==" + }, + "semver@7.6.3": { + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==" + }, + "serialize-javascript@6.0.2": { + "integrity": "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==", + "dependencies": [ + "randombytes" + ] + }, + "shebang-command@2.0.0": { + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dependencies": [ + "shebang-regex" + ] + }, + "shebang-regex@3.0.0": { + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==" + }, + "shelljs@0.8.5": { + "integrity": "sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow==", + "dependencies": [ + "glob@7.2.3", + "interpret", + "rechoir" + ] + }, + "shiki@1.22.2": { + "integrity": "sha512-3IZau0NdGKXhH2bBlUk4w1IHNxPh6A5B2sUpyY+8utLu2j/h1QpFkAaUA1bAMxOWWGtTWcAh531vnS4NJKS/lA==", + "dependencies": [ + "@shikijs/core", + "@shikijs/engine-javascript", + "@shikijs/engine-oniguruma", + "@shikijs/types", + "@shikijs/vscode-textmate", + "@types/hast" + ] + }, + "shx@0.3.4": { + "integrity": "sha512-N6A9MLVqjxZYcVn8hLmtneQWIJtp8IKzMP4eMnx+nqkvXoqinUPCbUFLp2UcWTEIUONhlk0ewxr/jaVGlc+J+g==", + "dependencies": [ + "minimist", + "shelljs" + ] + }, + "signal-exit@3.0.7": { + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==" + }, + "signal-exit@4.1.0": { + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==" + }, + "slash@3.0.0": { + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==" + }, + "smob@1.5.0": { + "integrity": "sha512-g6T+p7QO8npa+/hNx9ohv1E5pVCmWrVCUzUXJyLdMmftX6ER0oiWY/w9knEonLpnOp6b6FenKnMfR8gqwWdwig==" + }, + "source-map-js@1.2.1": { + "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==" + }, + "source-map-support@0.5.21": { + "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", + "dependencies": [ + "buffer-from", + "source-map@0.6.1" + ] + }, + "source-map@0.6.1": { + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" + }, + "source-map@0.7.4": { + "integrity": "sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==" + }, + "source-map@0.8.0-beta.0": { + "integrity": "sha512-2ymg6oRBpebeZi9UUNsgQ89bhx01TcTkmNTGnNO88imTmbSgy4nfujrgVEFKWpMTEGA11EDkTt7mqObTPdigIA==", + "dependencies": [ + "whatwg-url" + ] + }, + "space-separated-tokens@2.0.2": { + "integrity": "sha512-PEGlAwrG8yXGXRjW32fGbg66JAlOAwbObuqVoJpv/mRgoWDQfgH1wDPvtzWyUSNAXBGSk8h755YDbbcEy3SH2Q==" + }, + "stream-to-array@2.3.0": { + "integrity": "sha512-UsZtOYEn4tWU2RGLOXr/o/xjRBftZRlG3dEWoaHr8j4GuypJ3isitGbVyjQKAuMu+xbiop8q224TjiZWc4XTZA==", + "dependencies": [ + "any-promise" + ] + }, + "string-width@4.2.3": { + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dependencies": [ + "emoji-regex@8.0.0", + "is-fullwidth-code-point", + "strip-ansi@6.0.1" + ] + }, + "string-width@5.1.2": { + "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", + "dependencies": [ + "eastasianwidth", + "emoji-regex@9.2.2", + "strip-ansi@7.1.0" + ] + }, + "string_decoder@1.3.0": { + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "dependencies": [ + "safe-buffer" + ] + }, + "stringify-entities@4.0.4": { + "integrity": "sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg==", + "dependencies": [ + "character-entities-html4", + "character-entities-legacy" + ] + }, + "stringify-object@3.3.0": { + "integrity": "sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw==", + "dependencies": [ + "get-own-enumerable-property-symbols", + "is-obj", + "is-regexp" + ] + }, + "strip-ansi@6.0.1": { + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": [ + "ansi-regex@5.0.1" + ] + }, + "strip-ansi@7.1.0": { + "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", + "dependencies": [ + "ansi-regex@6.1.0" + ] + }, + "strip-bom@3.0.0": { + "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==" + }, + "strip-json-comments@2.0.1": { + "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==" + }, + "stylus-lookup@5.0.1": { + "integrity": "sha512-tLtJEd5AGvnVy4f9UHQMw4bkJJtaAcmo54N+ovQBjDY3DuWyK9Eltxzr5+KG0q4ew6v2EHyuWWNnHeiw/Eo7rQ==", + "dependencies": [ + "commander@10.0.1" + ] + }, + "sucrase@3.35.0": { + "integrity": "sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==", + "dependencies": [ + "@jridgewell/gen-mapping", + "commander@4.1.1", + "glob@10.4.5", + "lines-and-columns", + "mz", + "pirates", + "ts-interface-checker" + ] + }, + "supports-color@7.2.0": { + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dependencies": [ + "has-flag" + ] + }, + "supports-preserve-symlinks-flag@1.0.0": { + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==" + }, + "svelte-sonner@0.3.28_svelte@5.1.5__acorn@8.14.0": { + "integrity": "sha512-K3AmlySeFifF/cKgsYNv5uXqMVNln0NBAacOYgmkQStLa/UoU0LhfAACU6Gr+YYC8bOCHdVmFNoKuDbMEsppJg==", + "dependencies": [ + "svelte" + ] + }, + "svelte@5.1.5_acorn@8.14.0": { + "integrity": "sha512-AyYondx6wS0g8mmBMfwJVnOYYBswjBv6L4bc99awfbET2KozWvVwxe8NSN7fhx7Pgr7pOfOXIv7K8+Impc0OoQ==", + "dependencies": [ + "@ampproject/remapping", + "@jridgewell/sourcemap-codec", + "@types/estree", + "acorn", + "acorn-typescript", + "aria-query", + "axobject-query", + "esm-env", + "esrap", + "is-reference", + "locate-character", + "magic-string", + "zimmerframe" + ] + }, + "tapable@2.2.1": { + "integrity": "sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==" + }, + "tauri-api-adapter@0.3.8_typescript@5.6.3_rollup@4.24.3": { + "integrity": "sha512-HX6VMCWLzMIhTVuEaKbgLbQXR3YpIrzAgZGPv0Nge7zIn352G5vA/QJxQ9rlpFO4Y8S+Ro7VXR7DpVPMisTHaQ==", + "dependencies": [ + "@huakunshen/comlink", + "@rollup/plugin-alias", + "@rollup/plugin-typescript", + "@tauri-apps/api@2.0.3", + "@tauri-apps/plugin-dialog", + "@tauri-apps/plugin-fs", + "@tauri-apps/plugin-http", + "@tauri-apps/plugin-log", + "@tauri-apps/plugin-notification", + "@tauri-apps/plugin-os", + "@tauri-apps/plugin-shell", + "@tauri-apps/plugin-upload", + "rimraf", + "rollup", + "shx", + "tauri-plugin-clipboard-api", + "tauri-plugin-network-api", + "tauri-plugin-shellx-api", + "tauri-plugin-system-info-api", + "tsc-alias", + "typescript", + "valibot" + ] + }, + "tauri-plugin-clipboard-api@2.1.11_typescript@5.6.3": { + "integrity": "sha512-VNkGaVPPfRoHg7/rJBcWqsvLvn4/kNEOOlzqwyI9Qdf6g54B3mc31GLZdnq/HWtX0vZskw3J8b/EF9YkASDCBQ==", + "dependencies": [ + "@tauri-apps/api@2.0.1", + "valibot" + ] + }, + "tauri-plugin-network-api@2.0.3_typescript@5.6.3": { + "integrity": "sha512-aj/o315shIntam+vxVlgRJE8NnKcf0d93CQBK6zOb3Ia4V4HM+kYdFxg8Ji5jM5Yay8nq1HEJaqNIOIv7R1NpA==", + "dependencies": [ + "@tauri-apps/api@2.0.3", + "valibot" + ] + }, + "tauri-plugin-shellx-api@2.0.11": { + "integrity": "sha512-+FKIP1FBHdIQ6tASohww3MOf/8CDvYMYpPg9glO59h8TGVxTNP2ofiOEKLYk8M/o2H4tP7mxxca11QpDAT2LXw==", + "dependencies": [ + "@tauri-apps/api@2.0.3" + ] + }, + "tauri-plugin-system-info-api@2.0.6_typescript@5.6.3": { + "integrity": "sha512-7q8ieMK1bweMiLDbIWNNX3Dy7KNNwiB+WNEqKQutl0TKEGAb+efpy2SAzVLGu+i9tjDagCai8fZNuLLIpoQ9TA==", + "dependencies": [ + "@tauri-apps/api@2.0.3", + "valibot" + ] + }, + "terser@5.36.0": { + "integrity": "sha512-IYV9eNMuFAV4THUspIRXkLakHnV6XO7FEdtKjf/mDyrnqUg9LnlOn6/RwRvM9SZjR4GUq8Nk8zj67FzVARr74w==", + "dependencies": [ + "@jridgewell/source-map", + "acorn", + "commander@2.20.3", + "source-map-support" + ] + }, + "thenify-all@1.6.0": { + "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==", + "dependencies": [ + "thenify" + ] + }, + "thenify@3.3.1": { + "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==", + "dependencies": [ + "any-promise" + ] + }, + "tinyexec@0.3.1": { + "integrity": "sha512-WiCJLEECkO18gwqIp6+hJg0//p23HXp4S+gGtAKu3mI2F2/sXC4FvHvXvB0zJVVaTPhx1/tOwdbRsa1sOBIKqQ==" + }, + "tinyglobby@0.2.10_picomatch@4.0.2": { + "integrity": "sha512-Zc+8eJlFMvgatPZTl6A9L/yht8QqdmUNtURHaKZLmKBE12hNPSrqNkUp2cs3M/UKmNVVAMFQYSjYIVHDjW5zew==", + "dependencies": [ + "fdir", + "picomatch@4.0.2" + ] + }, + "to-regex-range@5.0.1": { + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dependencies": [ + "is-number" + ] + }, + "tr46@1.0.1": { + "integrity": "sha512-dTpowEjclQ7Kgx5SdBkqRzVhERQXov8/l9Ft9dVM9fmg0W0KQSVaXX9T4i6twCPNtYiZM53lpSSUAwJbFPOHxA==", + "dependencies": [ + "punycode" + ] + }, + "tree-kill@1.2.2": { + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==" + }, + "trim-lines@3.0.1": { + "integrity": "sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg==" + }, + "ts-graphviz@1.8.2": { + "integrity": "sha512-5YhbFoHmjxa7pgQLkB07MtGnGJ/yhvjmc9uhsnDBEICME6gkPf83SBwLDQqGDoCa3XzUMWLk1AU2Wn1u1naDtA==" + }, + "ts-interface-checker@0.1.13": { + "integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==" + }, + "tsc-alias@1.8.10": { + "integrity": "sha512-Ibv4KAWfFkFdKJxnWfVtdOmB0Zi1RJVxcbPGiCDsFpCQSsmpWyuzHG3rQyI5YkobWwxFPEyQfu1hdo4qLG2zPw==", + "dependencies": [ + "chokidar@3.6.0", + "commander@9.5.0", + "globby", + "mylas", + "normalize-path", + "plimit-lit" + ] + }, + "tsconfig-paths@4.2.0": { + "integrity": "sha512-NoZ4roiN7LnbKn9QqE1amc9DJfzvZXxF4xDavcOWt1BPkdx+m+0gJuPM+S0vCe7zTJMYUP0R8pO2XMr+Y8oLIg==", + "dependencies": [ + "json5", + "minimist", + "strip-bom" + ] + }, + "tslib@1.14.1": { + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" + }, + "tslib@2.8.0": { + "integrity": "sha512-jWVzBLplnCmoaTr13V9dYbiQ99wvZRd0vNWaDRg+aVYRcjDF3nDksxFDE/+fkXnKhpnUUkmx5pK/v8mCtLVqZA==" + }, + "tsup@8.3.5_typescript@5.6.3_esbuild@0.24.0": { + "integrity": "sha512-Tunf6r6m6tnZsG9GYWndg0z8dEV7fD733VBFzFJ5Vcm1FtlXB8xBD/rtrBi2a3YKEV7hHtxiZtW5EAVADoe1pA==", + "dependencies": [ + "bundle-require", + "cac", + "chokidar@4.0.1", + "consola", + "debug", + "esbuild", + "joycon", + "picocolors", + "postcss-load-config", + "resolve-from", + "rollup", + "source-map@0.8.0-beta.0", + "sucrase", + "tinyexec", + "tinyglobby", + "tree-kill", + "typescript" + ] + }, + "tsutils@3.21.0_typescript@5.6.3": { + "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", + "dependencies": [ + "tslib@1.14.1", + "typescript" + ] + }, + "typedoc@0.26.10_typescript@5.6.3": { + "integrity": "sha512-xLmVKJ8S21t+JeuQLNueebEuTVphx6IrP06CdV7+0WVflUSW3SPmR+h1fnWVdAR/FQePEgsSWCUHXqKKjzuUAw==", + "dependencies": [ + "lunr", + "markdown-it", + "minimatch@9.0.5", + "shiki", + "typescript", + "yaml" + ] + }, + "typescript@5.6.3": { + "integrity": "sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw==" + }, + "uc.micro@2.1.0": { + "integrity": "sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A==" + }, + "undici-types@5.26.5": { + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==" + }, + "undici-types@6.19.8": { + "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==" + }, + "unist-util-is@6.0.0": { + "integrity": "sha512-2qCTHimwdxLfz+YzdGfkqNlH0tLi9xjTnHddPmJwtIG9MGsdbutfTc4P+haPD7l7Cjxf/WZj+we5qfVPvvxfYw==", + "dependencies": [ + "@types/unist" + ] + }, + "unist-util-position@5.0.0": { + "integrity": "sha512-fucsC7HjXvkB5R3kTCO7kUjRdrS0BJt3M/FPxmHMBOm8JQi2BsHAHFsy27E0EolP8rp0NzXsJ+jNPyDWvOJZPA==", + "dependencies": [ + "@types/unist" + ] + }, + "unist-util-stringify-position@4.0.0": { + "integrity": "sha512-0ASV06AAoKCDkS2+xw5RXJywruurpbC4JZSm7nr7MOt1ojAzvyyaO+UxZf18j8FCF6kmzCZKcAgN/yu2gm2XgQ==", + "dependencies": [ + "@types/unist" + ] + }, + "unist-util-visit-parents@6.0.1": { + "integrity": "sha512-L/PqWzfTP9lzzEa6CKs0k2nARxTdZduw3zyh8d2NVBnsyvHjSX4TWse388YrrQKbvI8w20fGjGlhgT96WwKykw==", + "dependencies": [ + "@types/unist", + "unist-util-is" + ] + }, + "unist-util-visit@5.0.0": { + "integrity": "sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg==", + "dependencies": [ + "@types/unist", + "unist-util-is", + "unist-util-visit-parents" + ] + }, + "universalify@2.0.1": { + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==" + }, + "util-deprecate@1.0.2": { + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + }, + "valibot@0.40.0_typescript@5.6.3": { + "integrity": "sha512-XHKnaVtwHqxPwnGOsLrwka9CEaL7yNeLNp707OKv/bmT29GnPVdl6PxBOZ6BW7hF66/6QT6iVbOlnW7qVPmoKw==", + "dependencies": [ + "typescript" + ] + }, + "vfile-message@4.0.2": { + "integrity": "sha512-jRDZ1IMLttGj41KcZvlrYAaI3CfqpLpfpf+Mfig13viT6NKvRzWZ+lXz0Y5D60w6uJIBAOGq9mSHf0gktF0duw==", + "dependencies": [ + "@types/unist", + "unist-util-stringify-position" + ] + }, + "vfile@6.0.3": { + "integrity": "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q==", + "dependencies": [ + "@types/unist", + "vfile-message" + ] + }, + "walkdir@0.4.1": { + "integrity": "sha512-3eBwRyEln6E1MSzcxcVpQIhRG8Q1jLvEqRmCZqS3dsfXEDR/AhOF4d+jHg1qvDCpYaVRZjENPQyrVxAkQqxPgQ==" + }, + "wcwidth@1.0.1": { + "integrity": "sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==", + "dependencies": [ + "defaults" + ] + }, + "webidl-conversions@4.0.2": { + "integrity": "sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==" + }, + "whatwg-url@7.1.0": { + "integrity": "sha512-WUu7Rg1DroM7oQvGWfOiAK21n74Gg+T4elXEQYkOhtyLeWiJFoOGLXPKI/9gzIie9CtwVLm8wtw6YJdKyxSjeg==", + "dependencies": [ + "lodash.sortby", + "tr46", + "webidl-conversions" + ] + }, + "which@2.0.2": { + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dependencies": [ + "isexe" + ] + }, + "wrap-ansi@7.0.0": { + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dependencies": [ + "ansi-styles@4.3.0", + "string-width@4.2.3", + "strip-ansi@6.0.1" + ] + }, + "wrap-ansi@8.1.0": { + "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", + "dependencies": [ + "ansi-styles@6.2.1", + "string-width@5.1.2", + "strip-ansi@7.1.0" + ] + }, + "wrappy@1.0.2": { + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==" + }, + "y18n@5.0.8": { + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==" + }, + "yaml@2.6.0": { + "integrity": "sha512-a6ae//JvKDEra2kdi1qzCyrJW/WZCgFi8ydDV+eXExl95t+5R+ijnqHJbz9tmMh8FUjx3iv2fCQ4dclAQlO2UQ==" + }, + "yargs-parser@21.1.1": { + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==" + }, + "yargs@17.7.2": { + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dependencies": [ + "cliui", + "escalade", + "get-caller-file", + "require-directory", + "string-width@4.2.3", + "y18n", + "yargs-parser" + ] + }, + "zimmerframe@1.1.2": { + "integrity": "sha512-rAbqEGa8ovJy4pyBxZM70hg4pE6gDgaQ0Sl9M3enG3I0d6H4XSAM3GeNGLKnsBpuijUow064sf7ww1nutC5/3w==" + }, + "zwitch@2.0.4": { + "integrity": "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A==" + } + }, + "workspace": { + "packageJson": { + "dependencies": [ + "npm:@huakunshen/comlink@^4.4.1", + "npm:@rollup/plugin-json@^6.1.0", + "npm:@rollup/plugin-terser@~0.4.4", + "npm:@rollup/plugin-typescript@^11.1.6", + "npm:@tauri-apps/api@^2.0.3", + "npm:@tauri-apps/cli@^2.0.4", + "npm:@tauri-apps/plugin-deep-link@2", + "npm:@tauri-apps/plugin-dialog@^2.0.1", + "npm:@tauri-apps/plugin-fs@^2.0.1", + "npm:@tauri-apps/plugin-global-shortcut@2", + "npm:@tauri-apps/plugin-http@^2.0.1", + "npm:@tauri-apps/plugin-log@2", + "npm:@tauri-apps/plugin-notification@2", + "npm:@tauri-apps/plugin-os@2", + "npm:@tauri-apps/plugin-process@2.0.0", + "npm:@tauri-apps/plugin-shell@^2.0.1", + "npm:@tauri-apps/plugin-store@^2.1.0", + "npm:@tauri-apps/plugin-updater@2", + "npm:@tauri-apps/plugin-upload@2", + "npm:@types/bun@latest", + "npm:@types/lodash@^4.17.10", + "npm:@types/madge@^5.0.3", + "npm:@types/semver@^7.5.8", + "npm:comlink-stdio@~0.1.7", + "npm:comlink@^4.4.1", + "npm:fs-extra@^11.2.0", + "npm:lodash@^4.17.21", + "npm:madge@7", + "npm:minimatch@^10.0.1", + "npm:rimraf@^5.0.7", + "npm:rollup-plugin-typescript2@0.36", + "npm:rollup-plugin-visualizer@^5.12.0", + "npm:rollup@^4.24.0", + "npm:semver@^7.6.3", + "npm:svelte-sonner@~0.3.28", + "npm:tauri-api-adapter@0.3.8", + "npm:tauri-plugin-network-api@2.0.3", + "npm:tauri-plugin-shellx-api@^2.0.10", + "npm:tauri-plugin-system-info-api@2.0.6", + "npm:tsc-alias@^1.8.10", + "npm:tsup@^8.1.2", + "npm:typedoc@~0.26.4", + "npm:typescript@5", + "npm:valibot@0.40" + ] + } + } +} diff --git a/packages/api/jsr.json b/packages/api/jsr.json new file mode 100644 index 0000000..334de1a --- /dev/null +++ b/packages/api/jsr.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://jsr.io/schema/config-file.v1.json", + "name": "@kunkun/api", + "version": "0.0.27", + "license": "MIT", + "exports": { + ".": "./src/index.ts", + "./ui": "./src/ui/index.ts", + "./ui/iframe": "./src/ui/iframe/index.ts", + "./ui/worker": "./src/ui/worker/index.ts", + "./models": "./src/models/index.ts", + "./commands": "./src/commands/index.ts", + "./runtime/deno": "./src/runtime/deno.ts", + "./permissions": "./src/permissions/index.ts", + "./supabase": "./src/supabase/index.ts", + "./supabase/types": "./src/supabase/database.types.ts", + "./dev": "./src/dev/index.ts", + "./events": "./src/events.ts" + }, + "imports": { + "@hk/comlink-stdio": "jsr:@hk/comlink-stdio@^0.1.6" + } +} diff --git a/packages/api/package.json b/packages/api/package.json new file mode 100644 index 0000000..f171610 --- /dev/null +++ b/packages/api/package.json @@ -0,0 +1,74 @@ +{ + "name": "@kksh/api", + "version": "0.0.27", + "type": "module", + "exports": { + ".": "./src/index.ts", + "./ui": "./src/ui/index.ts", + "./ui/iframe": "./src/ui/iframe/index.ts", + "./ui/worker": "./src/ui/worker/index.ts", + "./models": "./src/models/index.ts", + "./commands": "./src/commands/index.ts", + "./runtime/deno": "./src/runtime/deno.ts", + "./permissions": "./src/permissions/index.ts", + "./dev": "./src/dev/index.ts", + "./events": "./src/events.ts", + "./supabase": "./src/supabase/index.ts", + "./supabase/types": "./src/supabase/database.types.ts", + "./package.json": "./package.json" + }, + "license": "MIT", + "scripts": { + "test": "bun test --coverage", + "gen:deno:types": "deno types > deno.d.ts", + "build:docs": "npx typedoc", + "dev": "bun --watch build.ts", + "build": "bun build.ts", + "prepare": "bun setup.ts", + "format": "prettier --write \"**/*.{ts,tsx,md,vue,json,yaml,yml}\"" + }, + "devDependencies": { + "@types/bun": "latest", + "@types/lodash": "^4.17.13", + "@types/madge": "^5.0.3", + "@types/node": "^22.8.7", + "@types/semver": "^7.5.8", + "fs-extra": "^11.2.0", + "madge": "^8.0.0", + "typedoc": "^0.26.11", + "typescript": "^5.0.0" + }, + "dependencies": { + "@hk/comlink-stdio": "npm:comlink-stdio@^0.1.7", + "@huakunshen/comlink": "^4.4.1", + "@tauri-apps/api": "^2.0.3", + "@tauri-apps/cli": "^2.0.4", + "@tauri-apps/plugin-deep-link": "^2.0.0", + "@tauri-apps/plugin-dialog": "^2.0.1", + "@tauri-apps/plugin-fs": "^2.0.1", + "@tauri-apps/plugin-global-shortcut": "^2.0.0", + "@tauri-apps/plugin-http": "^2.0.1", + "@tauri-apps/plugin-log": "^2.0.0", + "@tauri-apps/plugin-notification": "^2.0.0", + "@tauri-apps/plugin-os": "^2.0.0", + "@tauri-apps/plugin-process": "2.0.0", + "@tauri-apps/plugin-shell": "^2.0.1", + "@tauri-apps/plugin-store": "^2.1.0", + "@tauri-apps/plugin-updater": "^2.0.0", + "@tauri-apps/plugin-upload": "^2.0.0", + "comlink": "^4.4.1", + "lodash": "^4.17.21", + "minimatch": "^10.0.1", + "semver": "^7.6.3", + "svelte-sonner": "^0.3.28", + "tauri-api-adapter": "0.3.8", + "tauri-plugin-network-api": "2.0.4", + "tauri-plugin-shellx-api": "^2.0.11", + "tauri-plugin-system-info-api": "2.0.8", + "valibot": "^0.40.0" + }, + "files": [ + "src", + "dist" + ] +} diff --git a/packages/api/patch-version.ts b/packages/api/patch-version.ts new file mode 100644 index 0000000..6d408bc --- /dev/null +++ b/packages/api/patch-version.ts @@ -0,0 +1,13 @@ +import fs from "fs" +import { version } from "./package.json" + +const versionTsContent = fs.readFileSync("./src/version.ts", "utf-8") +const lines: string[] = [] +for (const line of versionTsContent.split("\n")) { + if (line.includes("export const version")) { + lines.push(`export const version = "${version}"`) + } else { + lines.push(line) + } +} +fs.writeFileSync("./src/version.ts", lines.join("\n")) diff --git a/packages/api/setup.ts b/packages/api/setup.ts new file mode 100644 index 0000000..d2fa437 --- /dev/null +++ b/packages/api/setup.ts @@ -0,0 +1,7 @@ +import { $ } from "bun" + +// Generate deno.d.ts under packages/api +let denoTypes = await $`deno types`.text() +// grep to filter out the line in denoTypes that contains "no-default-lib" +denoTypes = denoTypes.split("\n").filter((line) => !line.includes("no-default-lib")).join("\n") +Bun.write("deno.d.ts", denoTypes) diff --git a/packages/api/src/commands/apps.ts b/packages/api/src/commands/apps.ts new file mode 100644 index 0000000..f4ca8fe --- /dev/null +++ b/packages/api/src/commands/apps.ts @@ -0,0 +1,26 @@ +import { invoke } from "@tauri-apps/api/core" +import { AppInfo } from "../models" +import { generateJarvisPluginCommand } from "./common" + +export function getAllApps(): Promise { + return invoke(generateJarvisPluginCommand("get_applications")) +} + +export function refreshApplicationsList(): Promise { + return invoke(generateJarvisPluginCommand("refresh_applications_list")) +} + +export function refreshApplicationsListInBg(): Promise { + return invoke(generateJarvisPluginCommand("refresh_applications_list_in_bg")) +} + +// export function convertAppToTListItem(app: AppInfo): TListItem { +// return { +// title: app.name, +// value: app.app_desktop_path, +// description: "", +// type: "Application", +// icon: null, +// keywords: app.name.split(" "), +// }; +// } diff --git a/packages/api/src/commands/clipboard.ts b/packages/api/src/commands/clipboard.ts new file mode 100644 index 0000000..c0a4538 --- /dev/null +++ b/packages/api/src/commands/clipboard.ts @@ -0,0 +1,35 @@ +import { invoke } from "@tauri-apps/api/core" +import { array, literal, number, object, parse, string, union, type InferOutput } from "valibot" +import { generateJarvisPluginCommand } from "./common" + +export const ClipboardContentType = union([ + literal("Text"), + literal("Image"), + literal("Html"), + literal("Rtf") + // z.literal("File"), +]) +export type ClipboardContentType = InferOutput +export const ClipboardRecord = object({ + value: string(), + contentType: ClipboardContentType, + timestamp: number(), + text: string() +}) +export type ClipboardRecord = InferOutput +export const ClipboardRecords = array(ClipboardRecord) +export type ClipboardRecords = InferOutput + +export function addClipboardHistory(value: string) { + return invoke(generateJarvisPluginCommand("add_to_history"), { value }) +} + +export function getClipboardHistory() { + return invoke(generateJarvisPluginCommand("get_history")).then((records) => { + return parse(ClipboardRecords, records) + }) +} + +// export function setCandidateFilesForServer(files: string[]) { +// return invoke(generateJarvisPluginCommand("set_candidate_files"), { files }) +// } diff --git a/packages/api/src/commands/common.ts b/packages/api/src/commands/common.ts new file mode 100644 index 0000000..0815318 --- /dev/null +++ b/packages/api/src/commands/common.ts @@ -0,0 +1,5 @@ +export const JarvisPluginCommandPrefix = "plugin:jarvis" + +export function generateJarvisPluginCommand(command: string) { + return `${JarvisPluginCommandPrefix}|${command}` +} diff --git a/packages/api/src/commands/db.ts b/packages/api/src/commands/db.ts new file mode 100644 index 0000000..9f653a1 --- /dev/null +++ b/packages/api/src/commands/db.ts @@ -0,0 +1,338 @@ +import { invoke } from "@tauri-apps/api/core" +import { array, literal, optional, parse, safeParse, union, type InferOutput } from "valibot" +import { KUNKUN_EXT_IDENTIFIER } from "../constants" +import { CmdType, Ext, ExtCmd, ExtData } from "../models/extension" +import { convertDateToSqliteString, SQLSortOrder } from "../models/sql" +import { generateJarvisPluginCommand } from "./common" + +/* -------------------------------------------------------------------------- */ +/* Extension CRUD */ +/* -------------------------------------------------------------------------- */ +export function createExtension(ext: { + identifier: string + version: string + enabled?: boolean + path?: string + data?: any +}) { + return invoke(generateJarvisPluginCommand("create_extension"), ext) +} + +export function getAllExtensions() { + return invoke(generateJarvisPluginCommand("get_all_extensions")) +} + +export function getUniqueExtensionByIdentifier(identifier: string) { + return invoke( + generateJarvisPluginCommand("get_unique_extension_by_identifier"), + { + identifier + } + ) +} + +export function getUniqueExtensionByPath(path: string) { + return invoke(generateJarvisPluginCommand("get_unique_extension_by_path"), { + path + }) +} + +export function getAllExtensionsByIdentifier(identifier: string) { + return invoke(generateJarvisPluginCommand("get_all_extensions_by_identifier"), { + identifier + }) +} + +/** + * Use this function when you expect the extension to exist. Such as builtin extensions. + * @param identifier + * @returns + */ +export function getExtensionByIdentifierExpectExists(identifier: string): Promise { + return getUniqueExtensionByIdentifier(identifier).then((ext) => { + if (!ext) { + throw new Error(`Unexpexted Error: Extension ${identifier} not found`) + } + return ext + }) +} + +// TODO: clean this up +// export function deleteExtensionByIdentifier(identifier: string) { +// return invoke(generateJarvisPluginCommand("delete_extension_by_identifier"), { identifier }) +// } + +export function deleteExtensionByPath(path: string) { + return invoke(generateJarvisPluginCommand("delete_extension_by_path"), { path }) +} + +export function deleteExtensionByExtId(extId: string) { + return invoke(generateJarvisPluginCommand("delete_extension_by_ext_id"), { extId }) +} + +/* -------------------------------------------------------------------------- */ +/* Extension Command CRUD */ +/* -------------------------------------------------------------------------- */ +export function createCommand(data: { + extId: number + name: string + cmdType: CmdType + data: string + alias?: string + hotkey?: string + enabled?: boolean +}) { + return invoke(generateJarvisPluginCommand("create_command"), { + ...data, + enabled: data.enabled ?? false + }) +} + +export function getCommandById(cmdId: number) { + return invoke(generateJarvisPluginCommand("get_command_by_id"), { cmdId }) +} + +export function getCommandsByExtId(extId: number) { + return invoke(generateJarvisPluginCommand("get_commands_by_ext_id"), { extId }) +} + +export function deleteCommandById(cmdId: number) { + return invoke(generateJarvisPluginCommand("delete_command_by_id"), { cmdId }) +} + +export function updateCommandById(data: { + cmdId: number + name: string + cmdType: CmdType + data: string + alias?: string + hotkey?: string + enabled: boolean +}) { + return invoke(generateJarvisPluginCommand("update_command_by_id"), data) +} + +/* -------------------------------------------------------------------------- */ +/* Extension Data CRUD */ +/* -------------------------------------------------------------------------- */ +export const ExtDataField = union([literal("data"), literal("search_text")]) +export type ExtDataField = InferOutput + +function convertRawExtDataToExtData(rawData?: { + createdAt: string + updatedAt: string + data: null | string + searchText: null | string +}): ExtData | undefined { + if (!rawData) { + return rawData + } + const parsedRes = safeParse(ExtData, { + ...rawData, + createdAt: new Date(rawData.createdAt), + updatedAt: new Date(rawData.updatedAt), + data: rawData.data ?? undefined, + searchText: rawData.searchText ?? undefined + }) + if (parsedRes.success) { + return parsedRes.output + } else { + console.error("Extension Data Parse Failure", parsedRes.issues) + throw new Error("Fail to parse extension data") + } +} + +export function createExtensionData(data: { + extId: number + dataType: string + data: string + searchText?: string +}) { + return invoke(generateJarvisPluginCommand("create_extension_data"), data) +} + +export function getExtensionDataById(dataId: number) { + return invoke< + | (ExtData & { + createdAt: string + updatedAt: string + data: null | string + searchText: null | string + }) + | undefined + >(generateJarvisPluginCommand("get_extension_data_by_id"), { + dataId + }).then(convertRawExtDataToExtData) +} + +/** + * Fields option can be used to select optional fields. By default, if left empty, data and searchText are not returned. + * This is because data and searchText can be large and we don't want to return them by default. + * If you just want to get data ids in order to delete them, retrieving all data is not necessary. + * @param searchParams + */ +export async function searchExtensionData(searchParams: { + extId: number + searchExactMatch: boolean + dataId?: number + dataType?: string + searchText?: string + afterCreatedAt?: string + beforeCreatedAt?: string + limit?: number + orderByCreatedAt?: SQLSortOrder + orderByUpdatedAt?: SQLSortOrder + fields?: ExtDataField[] +}): Promise { + const fields = parse(optional(array(ExtDataField), []), searchParams.fields) + let items = await invoke< + (ExtData & { + createdAt: string + updatedAt: string + data: null | string + searchText: null | string + })[] + >(generateJarvisPluginCommand("search_extension_data"), { ...searchParams, fields }) + + return items.map(convertRawExtDataToExtData).filter((item) => item) as ExtData[] +} + +export function deleteExtensionDataById(dataId: number) { + return invoke(generateJarvisPluginCommand("delete_extension_data_by_id"), { dataId }) +} + +export function updateExtensionDataById(data: { + dataId: number + data: string + searchText?: string +}) { + return invoke(generateJarvisPluginCommand("update_extension_data_by_id"), data) +} + +/* -------------------------------------------------------------------------- */ +/* Built-in Extensions */ +/* -------------------------------------------------------------------------- */ +export function getExtClipboard() { + return getExtensionByIdentifierExpectExists(KUNKUN_EXT_IDENTIFIER.KUNKUN_CLIPBOARD_EXT_IDENTIFIER) +} +export function getExtQuickLinks() { + return getExtensionByIdentifierExpectExists( + KUNKUN_EXT_IDENTIFIER.KUNKUN_QUICK_LINKS_EXT_IDENTIFIER + ) +} +export function getExtRemote() { + return getExtensionByIdentifierExpectExists(KUNKUN_EXT_IDENTIFIER.KUNKUN_REMOTE_EXT_IDENTIFIER) +} +export function getExtScriptCmd() { + return getExtensionByIdentifierExpectExists( + KUNKUN_EXT_IDENTIFIER.KUNKUN_SCRIPT_CMD_EXT_IDENTIFIER + ) +} +export function getExtDev() { + return getExtensionByIdentifierExpectExists(KUNKUN_EXT_IDENTIFIER.KUNKUN_DEV_EXT_IDENTIFIER) +} + +/** + * Database API for extensions. + * Extensions shouldn't have full access to the database, they can only access their own data. + * When an extension is loaded, the main thread will create an instance of this class and + * expose it to the extension. + */ +export class JarvisExtDB { + extId: number + + constructor(extId: number) { + this.extId = extId + } + + async add(data: { data: string; dataType?: string; searchText?: string }) { + return createExtensionData({ + data: data.data, + dataType: data.dataType ?? "default", + searchText: data.searchText, + extId: this.extId + }) + } + + async delete(dataId: number): Promise { + // Verify if this data belongs to this extension + const d = await getExtensionDataById(dataId) + if (!d || d.extId !== this.extId) { + throw new Error("Extension Data not found") + } + return await deleteExtensionDataById(dataId) + } + + async search(searchParams: { + dataId?: number + fullTextSearch?: boolean + dataType?: string + searchText?: string + afterCreatedAt?: Date + beforeCreatedAt?: Date + limit?: number + orderByCreatedAt?: SQLSortOrder + orderByUpdatedAt?: SQLSortOrder + fields?: ExtDataField[] + }): Promise { + const beforeCreatedAt = searchParams.beforeCreatedAt + ? convertDateToSqliteString(searchParams.beforeCreatedAt) + : undefined + const afterCreatedAt = searchParams.afterCreatedAt + ? convertDateToSqliteString(searchParams.afterCreatedAt) + : undefined + return searchExtensionData({ + ...searchParams, + searchExactMatch: searchParams.fullTextSearch ?? true, + extId: this.extId, + beforeCreatedAt, + afterCreatedAt + }) + } + + /** + * Retrieve all data of this extension. + * Use `search()` method for more advanced search. + * @param options optional fields to retrieve. By default, data and searchText are not returned. + * @returns + */ + retrieveAll(options: { fields?: ExtDataField[] }): Promise { + return this.search({ fields: options.fields }) + } + + /** + * Retrieve all data of this extension by type. + * Use `search()` method for more advanced search. + * @param dataType + * @returns + */ + retrieveAllByType(dataType: string): Promise { + return this.search({ dataType }) + } + + /** + * Delete all data of this extension. + */ + deleteAll(): Promise { + return this.search({}) + .then((items) => { + return Promise.all(items.map((item) => this.delete(item.dataId))) + }) + .then(() => {}) + } + + /** + * Update data and searchText of this extension. + * @param dataId unique id of the data + * @param data + * @param searchText + * @returns + */ + async update(data: { dataId: number; data: string; searchText?: string }): Promise { + const d = await getExtensionDataById(data.dataId) + if (!d || d.extId !== this.extId) { + throw new Error("Extension Data not found") + } + return updateExtensionDataById(data) + } +} diff --git a/packages/api/src/commands/extension.ts b/packages/api/src/commands/extension.ts new file mode 100644 index 0000000..3acdc52 --- /dev/null +++ b/packages/api/src/commands/extension.ts @@ -0,0 +1,47 @@ +import { invoke } from "@tauri-apps/api/core" +import { ExtensionLabelMap } from "../models/extension" +import { generateJarvisPluginCommand } from "./common" + +export function isWindowLabelRegistered(label: string): Promise { + return invoke(generateJarvisPluginCommand("is_window_label_registered"), { label }) +} + +/** + * @param extensionPath + * @returns Window Label + */ +export function registerExtensionWindow(options: { + extensionPath: string + windowLabel?: string + dist?: string +}): Promise { + const { extensionPath, windowLabel, dist } = options + return invoke(generateJarvisPluginCommand("register_extension_window"), { + extensionPath, + windowLabel, + dist + }) +} + +export function unregisterExtensionWindow(label: string): Promise { + console.log("unregisterExtensionWindow", label) + return invoke(generateJarvisPluginCommand("unregister_extension_window"), { label }) +} + +export function registerExtensionSpawnedProcess(windowLabel: string, pid: number): Promise { + return invoke(generateJarvisPluginCommand("register_extension_spawned_process"), { + windowLabel, + pid + }) +} + +export function unregisterExtensionSpawnedProcess(windowLabel: string, pid: number): Promise { + return invoke(generateJarvisPluginCommand("unregister_extension_spawned_process"), { + windowLabel, + pid + }) +} + +export function getExtLabelMap(): Promise { + return invoke(generateJarvisPluginCommand("get_ext_label_map")) +} diff --git a/packages/api/src/commands/fileSearch.ts b/packages/api/src/commands/fileSearch.ts new file mode 100644 index 0000000..9917e09 --- /dev/null +++ b/packages/api/src/commands/fileSearch.ts @@ -0,0 +1,42 @@ +import { invoke } from "@tauri-apps/api/core" +import { + array, + boolean, + nullable, + number, + object, + optional, + parse, + string, + type InferOutput +} from "valibot" +import { generateJarvisPluginCommand } from "./common" + +export const FileSearchParams = object({ + locations: array(string()), + query: optional(string()), + ext: optional(string()), + depth: optional(number()), + limit: optional(number()), + hidden: optional(boolean(), false), + ignore_case: optional(boolean(), false), + file_size_greater: optional(number()), + file_size_smaller: optional(number()), + file_size_equal: optional(number()), + created_after: optional(number()), + created_before: optional(number()), + modified_after: optional(number()), + modified_before: optional(number()) +}) +export type FileSearchParams = InferOutput + +export function fileSearch( + searchParams: Omit & { + hidden?: boolean + ignore_case?: boolean + } +): Promise { + return invoke(generateJarvisPluginCommand("file_search"), { + searchParams: parse(FileSearchParams, searchParams) + }) +} diff --git a/packages/api/src/commands/fs.ts b/packages/api/src/commands/fs.ts new file mode 100644 index 0000000..0a01491 --- /dev/null +++ b/packages/api/src/commands/fs.ts @@ -0,0 +1,70 @@ +import { invoke } from "@tauri-apps/api/core" +import { generateJarvisPluginCommand } from "./common" + +export function pathExists(path: string): Promise { + return invoke(generateJarvisPluginCommand("path_exists"), { path }) +} + +/** + * This command is built into Jarvis App + * Used to decompress a tarball file + * @param path + * @param destinationFolder + * @param options + * @returns + */ +export function decompressTarball( + path: string, + destinationFolder: string, + options?: { + overwrite?: boolean + } +): Promise { + return invoke(generateJarvisPluginCommand("decompress_tarball"), { + path, + destinationFolder, + overwrite: options?.overwrite ?? false + }) +} + +/** + * Compress a given directory into a tarball file + * @param srcDir Directory to compress + * @param destFile destination file, should end with .tar.gz or .tgz + * @param options + * @returns + */ +export function compressTarball( + srcDir: string, + destFile: string, + options?: { + overwrite?: boolean + } +): Promise { + return invoke(generateJarvisPluginCommand("compress_tarball"), { + srcDir, + destFile, + overwrite: options?.overwrite ?? false + }) +} + +/** + * + * @param path Path of file to unzip + * @param destinationFolder where to unzip the file + * @param options use overwrite to overwrite existing files + * @returns + */ +export function unzip( + path: string, + destinationFolder: string, + options?: { + overwrite?: boolean + } +): Promise { + return invoke(generateJarvisPluginCommand("unzip"), { + path, + destinationFolder, + overwrite: options?.overwrite ?? false + }) +} diff --git a/packages/api/src/commands/index.ts b/packages/api/src/commands/index.ts new file mode 100644 index 0000000..4074dc0 --- /dev/null +++ b/packages/api/src/commands/index.ts @@ -0,0 +1,14 @@ +export * from "./apps" +export * from "./fs" +export * from "./server" +export * from "./system" +export * from "./tools" +export * from "./extension" +export * from "./store" +export * as db from "./db" +export { JarvisExtDB } from "./db" +export * from "./clipboard" +export * from "./fileSearch" +export * from "./utils" +export * as macSecurity from "./mac-security" +export * from "./mdns" diff --git a/packages/api/src/commands/mac-security.ts b/packages/api/src/commands/mac-security.ts new file mode 100644 index 0000000..f8a0b0e --- /dev/null +++ b/packages/api/src/commands/mac-security.ts @@ -0,0 +1,14 @@ +import { invoke } from "@tauri-apps/api/core" +import { generateJarvisPluginCommand } from "./common" + +export function verifyAuth(): Promise { + return invoke(generateJarvisPluginCommand("verify_auth")) +} + +export function requestScreenCaptureAccess(): Promise { + return invoke(generateJarvisPluginCommand("request_screen_capture_access")) +} + +export function checkScreenCaptureAccess(): Promise { + return invoke(generateJarvisPluginCommand("check_screen_capture_access")) +} diff --git a/packages/api/src/commands/mdns.ts b/packages/api/src/commands/mdns.ts new file mode 100644 index 0000000..8ab29de --- /dev/null +++ b/packages/api/src/commands/mdns.ts @@ -0,0 +1,7 @@ +import { invoke } from "@tauri-apps/api/core" +import type { MdnsPeers } from "../models/mdns" +import { generateJarvisPluginCommand } from "./common" + +export function getPeers(): Promise { + return invoke(generateJarvisPluginCommand("get_peers")) +} diff --git a/packages/api/src/commands/path.ts b/packages/api/src/commands/path.ts new file mode 100644 index 0000000..74617b8 --- /dev/null +++ b/packages/api/src/commands/path.ts @@ -0,0 +1,16 @@ +import { invoke } from "@tauri-apps/api/core" +import { generateJarvisPluginCommand } from "./common" + +/** + * @returns /extensions + */ +export function getDefaultExtensionsDir(): Promise { + return invoke(generateJarvisPluginCommand("get_default_extensions_dir")) +} + +/** + * @returns /extensions_storage + */ +export function getDefaultExtensionsStorageDir(): Promise { + return invoke(generateJarvisPluginCommand("get_default_extensions_storage_dir")) +} diff --git a/packages/api/src/commands/server.ts b/packages/api/src/commands/server.ts new file mode 100644 index 0000000..444020d --- /dev/null +++ b/packages/api/src/commands/server.ts @@ -0,0 +1,41 @@ +import { invoke } from "@tauri-apps/api/core" +import { appDataDir, join } from "@tauri-apps/api/path" +import { generateJarvisPluginCommand } from "./common" + +export function startServer(): Promise { + return invoke(generateJarvisPluginCommand("start_server")) +} + +export function stopServer(): Promise { + return invoke(generateJarvisPluginCommand("stop_server")) +} + +export function restartServer(): Promise { + return invoke(generateJarvisPluginCommand("restart_server")) +} + +export function serverIsRunning(): Promise { + return invoke(generateJarvisPluginCommand("server_is_running")) +} + +// TODO: clean this up +// export function setDevExtensionFolder(devExtFolder: string | null): Promise { +// return invoke(generateJarvisPluginCommand("set_dev_extension_folder"), { devExtFolder }) +// } + +// export function setExtensionFolder(extFolder: string | null): Promise { +// return invoke(generateJarvisPluginCommand("set_extension_folder"), { extFolder }) +// } + +// export function getExtensionFolder(): Promise { +// return invoke(generateJarvisPluginCommand("get_extension_folder")) +// return appDataDir().then((dir) => join(dir, "extensions")) +// } + +// export function getDevExtensionFolder(): Promise { +// return invoke(generateJarvisPluginCommand("get_dev_extension_folder")) +// } + +export function getServerPort() { + return invoke(generateJarvisPluginCommand("get_server_port")) +} diff --git a/packages/api/src/commands/store.ts b/packages/api/src/commands/store.ts new file mode 100644 index 0000000..aa026a5 --- /dev/null +++ b/packages/api/src/commands/store.ts @@ -0,0 +1,223 @@ +import { invoke } from "@tauri-apps/api/core" +import { listen, type UnlistenFn } from "@tauri-apps/api/event" +import { generateJarvisPluginCommand } from "./common" + +const storageCmdPrefix = `ext_store_wrapper_` + +function computeCommandName(command: string): string { + return generateJarvisPluginCommand(`${storageCmdPrefix}${command}`) +} + +interface ChangePayload { + path: string + key: string + value: T | null +} + +/** + * This Store is actually a wrapper over the tauri-plugin-store. Customized to be used with Jarvis Extensions, the APIs are exactly the same. + * A key-value store for Jarvis Extensions. Create a store in UI Extensions to store any data. + * filename is optional for the constructor if you only need one store file. + * If you plan to have multiple stores, e.g. one for settings, one for data, you can specify different filenames. + * @example + * ```ts + * const store = new JarvisStore("settings.bin"); + * await store.set("theme", "dark"); + * const theme = await store.get("theme"); + * console.log(theme); // dark + * ``` + */ +export class JarvisStore { + path: string + /** + * filename is optional if you only need one store file. + * If you plan to have multiple stores, e.g. one for settings, one for data, you can specify different filenames. + * @example + * ```ts + * const store = new JarvisStore("settings.bin"); + * await store.set("theme", "dark"); + * const theme = await store.get("theme"); + * console.log(theme); // dark + * ``` + * @param filename filename for the store. Defaults to `default.bin`. + */ + constructor(filename: string = "default.bin") { + this.path = filename + } + + /** + * Inserts a key-value pair into the store. + * + * @param key + * @param value + * @returns + */ + async set(key: string, value: unknown): Promise { + await invoke(computeCommandName("set"), { + path: this.path, + key, + value + }) + } + + /** + * Returns the value for the given `key` or `null` the key does not exist. + * + * @param key + * @returns + */ + async get(key: string): Promise { + return await invoke(computeCommandName("get"), { + path: this.path, + key + }) + } + + /** + * Returns `true` if the given `key` exists in the store. + * + * @param key + * @returns + */ + async has(key: string): Promise { + return await invoke(computeCommandName("has"), { + path: this.path, + key + }) + } + + /** + * Removes a key-value pair from the store. + * + * @param key + * @returns + */ + async delete(key: string): Promise { + return await invoke(computeCommandName("delete"), { + path: this.path, + key + }) + } + + /** + * Clears the store, removing all key-value pairs. + * + * Note: To clear the storage and reset it to it's `default` value, use `reset` instead. + * @returns + */ + async clear(): Promise { + await invoke(computeCommandName("clear"), { + path: this.path + }) + } + + /** + * Resets the store to it's `default` value. + * + * If no default value has been set, this method behaves identical to `clear`. + * @returns + */ + async reset(): Promise { + await invoke(computeCommandName("reset"), { + path: this.path + }) + } + + /** + * Returns a list of all key in the store. + * + * @returns + */ + async keys(): Promise { + return await invoke(computeCommandName("keys"), { + path: this.path + }) + } + + /** + * Returns a list of all values in the store. + * + * @returns + */ + async values(): Promise { + return await invoke(computeCommandName("values"), { + path: this.path + }) + } + + /** + * Returns a list of all entries in the store. + * + * @returns + */ + async entries(): Promise> { + return await invoke(computeCommandName("entries"), { + path: this.path + }) + } + + /** + * Returns the number of key-value pairs in the store. + * + * @returns + */ + async length(): Promise { + return await invoke(computeCommandName("length"), { + path: this.path + }) + } + + /** + * Attempts to load the on-disk state at the stores `path` into memory. + * + * This method is useful if the on-disk state was edited by the user and you want to synchronize the changes. + * + * Note: This method does not emit change events. + * @returns + */ + async load(): Promise { + await invoke(computeCommandName("load"), { + path: this.path + }) + } + + /** + * Saves the store to disk at the stores `path`. + * + * As the store is only persisted to disk before the apps exit, changes might be lost in a crash. + * This method lets you persist the store to disk whenever you deem necessary. + * @returns + */ + async save(): Promise { + await invoke(computeCommandName("save"), { + path: this.path + }) + } + + /** + * Listen to changes on a store key. + * @param key + * @param cb + * @returns A promise resolving to a function to unlisten to the event. + */ + async onKeyChange(key: string, cb: (value: T | null) => void): Promise { + return await listen>("store://change", (event) => { + if (event.payload.path === this.path && event.payload.key === key) { + cb(event.payload.value) + } + }) + } + + /** + * Listen to changes on the store. + * @param cb + * @returns A promise resolving to a function to unlisten to the event. + */ + async onChange(cb: (key: string, value: T | null) => void): Promise { + return await listen>("store://change", (event) => { + if (event.payload.path === this.path) { + cb(event.payload.key, event.payload.value) + } + }) + } +} diff --git a/packages/api/src/commands/system.ts b/packages/api/src/commands/system.ts new file mode 100644 index 0000000..953f7fb --- /dev/null +++ b/packages/api/src/commands/system.ts @@ -0,0 +1,312 @@ +import { invoke } from "@tauri-apps/api/core" +import { platform } from "@tauri-apps/plugin-os" +import { parse } from "valibot" +import { AppInfo, IconEnum, SysCommand } from "../models" +import { generateJarvisPluginCommand } from "./common" + +export function openTrash(): Promise { + return invoke(generateJarvisPluginCommand("open_trash")) +} + +export function emptyTrash(): Promise { + return invoke(generateJarvisPluginCommand("empty_trash")) +} + +export function shutdown(): Promise { + return invoke(generateJarvisPluginCommand("shutdown")) +} + +export function reboot(): Promise { + return invoke(generateJarvisPluginCommand("reboot")) +} + +export function sleep(): Promise { + return invoke(generateJarvisPluginCommand("sleep")) +} + +export function toggleSystemAppearance(): Promise { + return invoke(generateJarvisPluginCommand("toggle_system_appearance")) +} + +export function showDesktop(): Promise { + return invoke(generateJarvisPluginCommand("show_desktop")) +} + +export function quitAllApps(): Promise { + return invoke(generateJarvisPluginCommand("quit_app_apps")) +} + +export function sleepDisplays(): Promise { + return invoke(generateJarvisPluginCommand("sleep_displays")) +} + +export function setVolume(percentage: number): Promise { + return invoke(generateJarvisPluginCommand("set_volume"), { percentage }) +} + +export function setVolumeTo0(): Promise { + return setVolume(0) +} + +export function setVolumeTo25(): Promise { + return setVolume(25) +} + +export function setVolumeTo50(): Promise { + return setVolume(50) +} + +export function setVolumeTo75(): Promise { + return setVolume(75) +} + +export function setVolumeTo100(): Promise { + return setVolume(100) +} + +export function turnVolumeUp(): Promise { + return invoke(generateJarvisPluginCommand("turn_volume_up")) +} + +export function turnVolumeDown(): Promise { + return invoke(generateJarvisPluginCommand("turn_volume_down")) +} + +export function toggleStageManager(): Promise { + return invoke(generateJarvisPluginCommand("toggle_stage_manager")) +} + +export function toggleBluetooth(): Promise { + return invoke(generateJarvisPluginCommand("toggle_bluetooth")) +} + +export function toggleHiddenFiles(): Promise { + return invoke(generateJarvisPluginCommand("toggle_hidden_files")) +} + +export function ejectAllDisks(): Promise { + return invoke(generateJarvisPluginCommand("eject_all_disks")) +} + +export function logoutUser(): Promise { + return invoke(generateJarvisPluginCommand("logout_user")) +} + +export function toggleMute(): Promise { + return invoke(generateJarvisPluginCommand("toggle_mute")) +} + +export function mute(): Promise { + return invoke(generateJarvisPluginCommand("mute")) +} + +export function unmute(): Promise { + return invoke(generateJarvisPluginCommand("unmute")) +} + +export function getFrontmostApp(): Promise { + return invoke(generateJarvisPluginCommand("get_frontmost_app")).then((app) => parse(AppInfo, app)) +} + +export function hideAllAppsExceptFrontmost(): Promise { + return invoke(generateJarvisPluginCommand("hide_all_apps_except_frontmost")) +} + +export function getSelectedFilesInFileExplorer(): Promise { + return invoke(generateJarvisPluginCommand("get_selected_files_in_file_explorer")) +} + +export const rawSystemCommands = [ + { + name: "Open Trash", + icon: "uil:trash", + confirmRequired: false, + function: openTrash, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Empty Trash", + icon: "uil:trash", + confirmRequired: true, + function: emptyTrash, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Shutdown", + icon: "mdi:shutdown", + confirmRequired: true, + function: shutdown, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Reboot", + icon: "mdi:restart", + confirmRequired: true, + function: reboot, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Sleep", + icon: "carbon:asleep", + confirmRequired: false, + function: sleep, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Toggle System Appearance", + icon: "line-md:light-dark", + confirmRequired: false, + function: toggleSystemAppearance, + platforms: ["macos"] + }, + { + name: "Show Desktop", + icon: "bi:window-desktop", + confirmRequired: false, + function: showDesktop, + platforms: ["macos"] + }, + { + name: "Quit App", + icon: "charm:cross", + confirmRequired: false, + function: quitAllApps, + platforms: [] + // platforms: ["macos"] + }, + { + name: "Sleep Displays", + icon: "solar:display-broken", + confirmRequired: false, + function: sleepDisplays, + platforms: ["macos"] + }, + { + name: "Set Volume to 0%", + icon: "flowbite:volume-mute-outline", + confirmRequired: false, + function: setVolumeTo0, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Set Volume to 25%", + icon: "flowbite:volume-down-solid", + confirmRequired: false, + function: setVolumeTo25, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Set Volume to 50%", + icon: "flowbite:volume-down-solid", + confirmRequired: false, + function: setVolumeTo50, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Set Volume to 75%", + icon: "flowbite:volume-down-solid", + confirmRequired: false, + function: setVolumeTo75, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Set Volume to 100%", + icon: "flowbite:volume-up-solid", + confirmRequired: false, + function: setVolumeTo100, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Turn Volume Up", + icon: "flowbite:volume-down-solid", + confirmRequired: false, + function: turnVolumeUp, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Turn Volume Down", + icon: "flowbite:volume-down-outline", + confirmRequired: false, + function: turnVolumeDown, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Toggle Mute", + icon: "flowbite:volume-down-outline", + confirmRequired: false, + function: toggleMute, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Mute", + icon: "flowbite:volume-mute-solid", + confirmRequired: false, + function: mute, + platforms: ["macos", "linux"] + }, + { + name: "Unmute", + icon: "flowbite:volume-mute-solid", + confirmRequired: false, + function: unmute, + platforms: ["macos", "linux"] + }, + { + name: "Toggle Stage Manager", + icon: "material-symbols:dashboard", + confirmRequired: false, + function: toggleStageManager, + platforms: [] + }, + { + name: "Toggle Bluetooth", + icon: "material-symbols:bluetooth", + confirmRequired: false, + function: toggleBluetooth, + platforms: [] + }, + { + name: "Toggle Hidden Files", + icon: "mdi:hide", + confirmRequired: false, + function: toggleHiddenFiles, + platforms: [] + }, + { + name: "Eject All Disks", + icon: "ph:eject-fill", + confirmRequired: true, + function: ejectAllDisks, + platforms: ["macos"] + }, + { + name: "Log Out User", + icon: "ic:baseline-logout", + confirmRequired: false, + function: logoutUser, + platforms: ["macos", "linux", "windows"] + }, + { + name: "Hide All Apps Except Frontmost", + icon: "mdi:hide", + confirmRequired: false, + function: hideAllAppsExceptFrontmost, + platforms: [] + } +] + +export async function getSystemCommands(): Promise { + return rawSystemCommands + .filter(async (cmd) => cmd.platforms.includes(platform())) // Filter out system commands that are not supported on the current platform + .map((cmd) => ({ + name: cmd.name, + value: "system-cmd" + cmd.name.split(" ").join("-").toLowerCase(), + icon: { + value: cmd.icon, + type: IconEnum.Iconify + }, + keywords: cmd.name.split(" "), + function: cmd.function, + confirmRequired: cmd.confirmRequired + })) +} diff --git a/packages/api/src/commands/tools.ts b/packages/api/src/commands/tools.ts new file mode 100644 index 0000000..277a80a --- /dev/null +++ b/packages/api/src/commands/tools.ts @@ -0,0 +1,26 @@ +import { invoke } from "@tauri-apps/api/core" +import { generateJarvisPluginCommand } from "./common" + +export function openDevTools(): Promise { + return invoke(generateJarvisPluginCommand("open_devtools")) +} + +export function closeDevTools(): Promise { + return invoke(generateJarvisPluginCommand("close_devtools")) +} + +export function toggleDevTools(): Promise { + return invoke(generateJarvisPluginCommand("toggle_devtools")) +} + +export function isDevToolsOpen(): Promise { + return invoke(generateJarvisPluginCommand("is_devtools_open")) +} + +/** + * Check if the app is running in development mode. + * @returns true if the app is running in development mode, false otherwise + */ +export function appIsDev(): Promise { + return invoke(generateJarvisPluginCommand("app_is_dev")) +} diff --git a/packages/api/src/commands/utils.ts b/packages/api/src/commands/utils.ts new file mode 100644 index 0000000..ac8bc78 --- /dev/null +++ b/packages/api/src/commands/utils.ts @@ -0,0 +1,6 @@ +import { invoke } from "@tauri-apps/api/core" +import { generateJarvisPluginCommand } from "./common" + +export function plistToJson(plistContent: string) { + return invoke(generateJarvisPluginCommand("plist_to_json"), { plistContent }) +} diff --git a/packages/api/src/constants.ts b/packages/api/src/constants.ts new file mode 100644 index 0000000..a3b8789 --- /dev/null +++ b/packages/api/src/constants.ts @@ -0,0 +1,18 @@ +export enum KUNKUN_EXT_IDENTIFIER { + KUNKUN_CLIPBOARD_EXT_IDENTIFIER = "sh.kunkun.ext.clipboard", + KUNKUN_QUICK_LINKS_EXT_IDENTIFIER = "sh.kunkun.ext.quick-links", + KUNKUN_REMOTE_EXT_IDENTIFIER = "sh.kunkun.ext.remote", + KUNKUN_SCRIPT_CMD_EXT_IDENTIFIER = "sh.kunkun.ext.script-cmd", + KUNKUN_DEV_EXT_IDENTIFIER = "sh.kunkun.ext.dev" +} + +export const KUNKUN_DESKTOP_APP_SERVER_PORTS = [1566, 1567, 1568, 9559, 9560, 9561] +export const DESKTOP_SERVICE_NAME = "Kunkun" + +/* -------------------------------------------------------------------------- */ +/* Deep Link */ +/* -------------------------------------------------------------------------- */ +export const DEEP_LINK_PATH_OPEN = "kunkun://open" +export const DEEP_LINK_PATH_STORE = "kunkun://store" +export const DEEP_LINK_PATH_REFRESH_DEV_EXTENSION = "kunkun://refresh-dev-extension" +export const DEEP_LINK_PATH_AUTH_CONFIRM = "kunkun://auth/confirm" diff --git a/packages/api/src/dev/index.ts b/packages/api/src/dev/index.ts new file mode 100644 index 0000000..4a44663 --- /dev/null +++ b/packages/api/src/dev/index.ts @@ -0,0 +1,83 @@ +import { exec } from "child_process" +import os from "node:os" +import { + DEEP_LINK_PATH_REFRESH_DEV_EXTENSION, + DESKTOP_SERVICE_NAME, + KUNKUN_DESKTOP_APP_SERVER_PORTS +} from "../constants" + +export function checkLocalKunkunService(port: number): Promise { + return fetch(`http://localhost:${port}/info`) + .then((res) => { + if (!res.ok) { + return false + } + return res.json() + }) + .then((data) => { + return data["service_name"] === DESKTOP_SERVICE_NAME + }) + .catch((err) => { + // fetch fail, i.e. server not on this port + return false + }) +} + +export async function findLocalhostKunkunPorts(): Promise { + const onlinePorts = [] + for (const port of KUNKUN_DESKTOP_APP_SERVER_PORTS) { + const online = await checkLocalKunkunService(port) + if (online) { + onlinePorts.push(port) + } + } + return onlinePorts +} + +export async function refreshTemplateWorkerExtensionViaServer() { + const ports = await findLocalhostKunkunPorts() + console.log("Kunkun ports", ports) + if (ports.length === 0) { + console.error("Failed to find localhost kunkun ports") + return + } else if (ports.length > 1) { + console.warn("Found multiple localhost kunkun ports", ports) + console.warn("Will Refresh Every Instance") + } + for (const port of ports) { + fetch(`http://localhost:${port}/refresh-worker-extension`, { method: "POST" }).catch((err) => { + console.error("Failed to send refresh worker extension request", err) + }) + } +} + +export async function refreshTemplateWorkerExtensionViaDeepLink() { + console.log("Send Refresh Worker Extension Request") + + const platform = await os.platform() + try { + switch (platform) { + case "darwin": + exec(`open ${DEEP_LINK_PATH_REFRESH_DEV_EXTENSION}`) + break + case "win32": + exec(`start ${DEEP_LINK_PATH_REFRESH_DEV_EXTENSION}`) + break + case "linux": + exec(`xdg-open ${DEEP_LINK_PATH_REFRESH_DEV_EXTENSION}`) + break + } + } catch (error) { + console.error("Failed to refresh worker extension:", error) + } +} + +export const refreshTemplateWorkerExtension = refreshTemplateWorkerExtensionViaServer + +export function kununWorkerTemplateExtensionRollupPlugin() { + return { + async writeBundle() { + await refreshTemplateWorkerExtensionViaDeepLink() + } + } +} diff --git a/packages/api/src/events.ts b/packages/api/src/events.ts new file mode 100644 index 0000000..c2654bc --- /dev/null +++ b/packages/api/src/events.ts @@ -0,0 +1,31 @@ +import { emitTo, listen, type EventCallback, type UnlistenFn } from "@tauri-apps/api/event" + +/* ------------------------------- Event Names ------------------------------ */ +export const RECORD_EXTENSION_PROCESS_EVENT = "record-extension-process" +export interface IRecordExtensionProcessEvent { + windowLabel: string + pid: number +} + +/** + * listen to RECORD_EXTENSION_PROCESS_EVENT + * This event is emitted when an extension spawns a process. Processes won't be cleaned up automatically + * Extensions should clean up the processes when extension quits, but there is no way to guarantee this. + * So we need to record the processes spawned from extensions and clean up the processes when extension quits. + * @param callback + * @returns + */ +export function listenToRecordExtensionProcessEvent( + callback: EventCallback +): Promise { + return listen(RECORD_EXTENSION_PROCESS_EVENT, callback) +} + +export const KILL_PROCESS_EVENT = "kunkun://kill-process" +export function emitKillProcessEvent(pid: number) { + return emitTo("main", KILL_PROCESS_EVENT, { pid }) +} + +export function listenToKillProcessEvent(cb: EventCallback<{ pid: number }>) { + return listen<{ pid: number }>(KILL_PROCESS_EVENT, cb) +} diff --git a/packages/api/src/index.ts b/packages/api/src/index.ts new file mode 100644 index 0000000..a746aed --- /dev/null +++ b/packages/api/src/index.ts @@ -0,0 +1,9 @@ +export { + version, + breakingChangesVersionCheckpoints, + isVersionBetween, + isCompatible +} from "./version" +export { constructExtensionSupportDir } from "./ui/server/path" +export * from "./constants" +export { TauriShellStdio } from "./ui/api/shell" diff --git a/packages/api/src/models/__tests__/schema.test.ts b/packages/api/src/models/__tests__/schema.test.ts new file mode 100644 index 0000000..5c4c724 --- /dev/null +++ b/packages/api/src/models/__tests__/schema.test.ts @@ -0,0 +1,31 @@ +import fs from "fs" +import path from "path" +import { PACKAGES_PATHS } from "@kksh/ci" +import { expect, test } from "bun:test" +import { safeParse } from "valibot" +import { ExtPackageJson } from "../manifest" + +test("Load and parse every extension in this repo", () => { + // iterate over all extensions in extensionsDir, parse package.json + ;[PACKAGES_PATHS.EXTENSIONS, PACKAGES_PATHS.TEMPLATES].forEach((dir) => { + fs.readdirSync(dir).forEach((extDirName) => { + const extDir = path.join(dir, extDirName) + const packageJsonPath = path.join(extDir, "package.json") + console.log(packageJsonPath) + if (!fs.existsSync(packageJsonPath)) { + return + } + // read package.json + + const pkgJsonContent = fs.readFileSync(packageJsonPath, "utf-8") + const pkgJson = JSON.parse(pkgJsonContent) + // validate package.json + // const result = parse(ExtPackageJson, pkgJson) + const parse = safeParse(ExtPackageJson, pkgJson) + if (parse.issues) { + console.log(parse.issues) + } + expect(parse.success).toBe(true) + }) + }) +}) diff --git a/packages/api/src/models/apps.ts b/packages/api/src/models/apps.ts new file mode 100644 index 0000000..9a7a722 --- /dev/null +++ b/packages/api/src/models/apps.ts @@ -0,0 +1,9 @@ +import { nullable, object, string, type InferOutput } from "valibot" + +export const AppInfo = object({ + name: string(), + icon_path: nullable(string()), + app_path_exe: nullable(string()), + app_desktop_path: string() +}) +export type AppInfo = InferOutput diff --git a/packages/api/src/models/constants.ts b/packages/api/src/models/constants.ts new file mode 100644 index 0000000..b69cf8c --- /dev/null +++ b/packages/api/src/models/constants.ts @@ -0,0 +1,42 @@ +import { enum_, type InferOutput } from "valibot" + +export enum NodeNameEnum { + List = "List", + ListItem = "ListItem", + ListItemDetail = "ListItemDetail", + ListItemAccessory = "ListItemAccessory", + ListSection = "ListSection", + ListItemDetailMetadata = "ListItemDetailMetadata", + ListItemDetailMetadataLabel = "ListItemDetailMetadataLabel", + ListItemDetailMetadataLink = "ListItemDetailMetadataLink", + ListItemDetailMetadataTagList = "ListItemDetailMetadataTagList", + ListItemDetailMetadataTagListItem = "ListItemDetailMetadataTagListItem", + ListItemDetailMetadataSeparator = "ListItemDetailMetadataSeparator", + Icon = "Icon", + EmptyView = "EmptyView", + Dropdown = "Dropdown", + DropdownSection = "DropdownSection", + DropdownItem = "DropdownItem", + /* --------------------------------- Action --------------------------------- */ + ActionPanel = "ActionPanel", + Action = "Action", + ActionPanelSection = "ActionPanelSection", + ActionPanelSubmenu = "ActionPanelSubmenu", + /* --------------------------------- Content -------------------------------- */ + Markdown = "Markdown" +} +export const NodeName = enum_(NodeNameEnum) +export type NodeName = InferOutput + +export enum FormNodeNameEnum { + Base = "Base", + Number = "Number", + Select = "Select", + Boolean = "Boolean", + Input = "Input", + Date = "Date", + Array = "Array", + Form = "Form" +} +export const FormNodeName = enum_(FormNodeNameEnum) +export type FormNodeName = InferOutput diff --git a/packages/api/src/models/extension.ts b/packages/api/src/models/extension.ts new file mode 100644 index 0000000..0b1a150 --- /dev/null +++ b/packages/api/src/models/extension.ts @@ -0,0 +1,83 @@ +import { + any, + array, + boolean, + date, + enum_, + function_, + nullable, + number, + object, + optional, + record, + string, + type InferOutput +} from "valibot" +import { Icon } from "./icon" + +/** + * Map window label to extension + */ +export const ExtensionLabelMap = record( + string("Window label"), + object({ + path: string("Path to the extension"), + processes: array(number()), + dist: optional(nullable(string())) + }) +) +export type ExtensionLabelMap = InferOutput + +export const Ext = object({ + extId: number(), + identifier: string(), + version: string(), + enabled: boolean(), + installed_at: string(), + path: nullable(string()), + data: nullable(any()) +}) +export type Ext = InferOutput + +export enum CmdTypeEnum { + HeadlessWorker = "headless_worker", + UiWorker = "ui_worker", + UiIframe = "ui_iframe", + QuickLink = "quick_link", + Remote = "remote" +} + +export const CmdType = enum_(CmdTypeEnum) +export type CmdType = InferOutput +export const ExtCmd = object({ + cmdId: number(), + extId: number(), + name: string(), + type: CmdType, + data: string(), + alias: optional(string()), + hotkey: optional(string()), + enabled: boolean() +}) +export type ExtCmd = InferOutput + +export const ExtData = object({ + dataId: number(), + extId: number(), + dataType: string(), + data: optional(string()), + searchText: optional(string()), + createdAt: date(), + updatedAt: date() +}) +export type ExtData = InferOutput + +export const SysCommand = object({ + name: string(), + value: string(), + icon: nullable(Icon), + keywords: nullable(array(string())), + function: function_(), + confirmRequired: boolean() +}) +export type SysCommand = InferOutput diff --git a/packages/api/src/models/icon.ts b/packages/api/src/models/icon.ts new file mode 100644 index 0000000..6639ba5 --- /dev/null +++ b/packages/api/src/models/icon.ts @@ -0,0 +1,26 @@ +import { enum_, literal, object, string, type InferOutput } from "valibot" +import { NodeName, NodeNameEnum } from "./constants" + +/* -------------------------------------------------------------------------- */ +/* Icon */ +/* -------------------------------------------------------------------------- */ +export enum IconEnum { + Iconify = "iconify", + RemoteUrl = "remote-url", + Svg = "svg", + Base64PNG = "base64-png", + Text = "text" +} +export const IconType = enum_(IconEnum) +export type IconType = InferOutput + +export const Icon = object({ + type: IconType, + value: string() +}) +export type Icon = InferOutput +export const IconNode = object({ + nodeName: NodeName, + ...Icon.entries +}) +export type IconNode = InferOutput diff --git a/packages/api/src/models/index.ts b/packages/api/src/models/index.ts new file mode 100644 index 0000000..12fddbf --- /dev/null +++ b/packages/api/src/models/index.ts @@ -0,0 +1,11 @@ +export * as List from "../ui/worker/schema/list" +export * as Action from "../ui/worker/schema/action" +export * from "./constants" +export * from "./icon" +export * from "./apps" +export * from "./extension" +export * from "./sql" +export * from "./manifest" +export * from "./styles" +export * from "./mdns" +export { AllKunkunPermission, SystemPermissionSchema } from "../permissions" diff --git a/packages/api/src/models/manifest.ts b/packages/api/src/models/manifest.ts new file mode 100644 index 0000000..056267d --- /dev/null +++ b/packages/api/src/models/manifest.ts @@ -0,0 +1,188 @@ +import { FsPermissionSchema } from "tauri-api-adapter/permissions" +import { + array, + boolean, + enum_, + literal, + nullable, + number, + object, + optional, + string, + union, + type InferOutput +} from "valibot" +import { + AllKunkunPermission, + FsPermissionScopedSchema, + KunkunFsPermissionSchema, + KunkunManifestPermission, + OpenPermissionScopedSchema, + ShellPermissionScopedSchema +} from "../permissions" +import { CmdType } from "./extension" +import { Icon } from "./icon" + +export enum OSPlatformEnum { + linux = "linux", + macos = "macos", + windows = "windows" +} + +export const OSPlatform = enum_(OSPlatformEnum) +export type OSPlatform = InferOutput +const allPlatforms = Object.values(OSPlatformEnum) +export const TriggerCmd = object({ + type: union([literal("text"), literal("regex")]), + value: string() +}) +export type TriggerCmd = InferOutput +export enum TitleBarStyleEnum { + "visible" = "visible", + "transparent" = "transparent", + "overlay" = "overlay" +} +export const TitleBarStyle = enum_(TitleBarStyleEnum) +// JS new WebViewWindow only accepts lowercase, while manifest loaded from Rust is capitalized. I run toLowerCase() on the value before passing it to the WebViewWindow. +// This lowercase title bar style schema is used to validate and set the type so TypeScript won't complaint +// export const TitleBarStyleAllLower = z.enum(["visible", "transparent", "overlay"]); +// export type TitleBarStyleAllLower = z.infer; +export const WindowConfig = object({ + center: optional(nullable(boolean())), + x: optional(nullable(number())), + y: optional(nullable(number())), + width: optional(nullable(number())), + height: optional(nullable(number())), + minWidth: optional(nullable(number())), + minHeight: optional(nullable(number())), + maxWidth: optional(nullable(number())), + maxHeight: optional(nullable(number())), + resizable: optional(nullable(boolean())), + title: optional(nullable(string())), + fullscreen: optional(nullable(boolean())), + focus: optional(nullable(boolean())), + transparent: optional(nullable(boolean())), + maximized: optional(nullable(boolean())), + visible: optional(nullable(boolean())), + decorations: optional(nullable(boolean())), + alwaysOnTop: optional(nullable(boolean())), + alwaysOnBottom: optional(nullable(boolean())), + contentProtected: optional(nullable(boolean())), + skipTaskbar: optional(nullable(boolean())), + shadow: optional(nullable(boolean())), + // theme: optional(nullable(union([literal("light"), literal("dark")]))), // changing theme of one window will change theme of all windows + titleBarStyle: optional(nullable(TitleBarStyle)), + hiddenTitle: optional(nullable(boolean())), + tabbingIdentifier: optional(nullable(string())), + maximizable: optional(nullable(boolean())), + minimizable: optional(nullable(boolean())), + closable: optional(nullable(boolean())), + parent: optional(nullable(string())), + visibleOnAllWorkspaces: optional(nullable(boolean())) +}) +export type WindowConfig = InferOutput +export const CustomUiCmd = object({ + type: optional(CmdType, CmdType.enum.UiIframe), + main: string("HTML file to load, e.g. dist/index.html"), + dist: string("Dist folder to load, e.g. dist, build, out"), + description: optional(nullable(string("Description of the Command"), ""), ""), + devMain: string("URL to load in development to support live reload, e.g. http://localhost:5173/"), + name: string("Name of the command"), + window: optional(nullable(WindowConfig)), + cmds: array(TriggerCmd, "Commands to trigger the UI"), + icon: optional(Icon), + platforms: optional( + nullable( + array(OSPlatform, "Platforms available on. Leave empty for all platforms."), + allPlatforms + ), + allPlatforms + ) +}) +export type CustomUiCmd = InferOutput + +export const TemplateUiCmd = object({ + type: optional(CmdType, CmdType.enum.UiWorker), + main: string(), + name: string(), + description: optional(nullable(string("Description of the Command"), ""), ""), + window: optional(nullable(WindowConfig)), + cmds: array(TriggerCmd), + icon: optional(Icon), + platforms: optional( + nullable( + array(OSPlatform, "Platforms available on. Leave empty for all platforms."), + allPlatforms + ), + allPlatforms + ) +}) +export type TemplateUiCmd = InferOutput +export const PermissionUnion = union([ + KunkunManifestPermission, + FsPermissionScopedSchema, + OpenPermissionScopedSchema, + ShellPermissionScopedSchema +]) +export type PermissionUnion = InferOutput +export const KunkunExtManifest = object({ + name: string("Name of the extension (Human Readable)"), + shortDescription: string("Description of the extension (Will be displayed in store)"), + longDescription: string("Long description of the extension (Will be displayed in store)"), + identifier: string( + "Unique identifier for the extension, must be the same as extension folder name" + ), + icon: Icon, + permissions: array( + PermissionUnion, + "Permissions Declared by the extension. e.g. clipboard-all. Not declared APIs will be blocked." + ), + demoImages: array(string("Demo images for the extension")), + customUiCmds: array(CustomUiCmd, "Custom UI Commands"), + templateUiCmds: array(TemplateUiCmd, "Template UI Commands") +}) +export type KunkunExtManifest = InferOutput + +const Person = union([ + object({ + name: string("GitHub Username"), + email: string("Email of the person"), + url: optional(nullable(string("URL of the person"))) + }), + string("GitHub Username") +]) + +export const ExtPackageJson = object({ + name: string("Package name for the extension (just a regular npm package name)"), + version: string("Version of the extension"), + author: optional(Person), + draft: optional(boolean("Whether the extension is a draft, draft will not be published")), + contributors: optional(array(Person, "Contributors of the extension")), + repository: optional( + union([ + string("URL of the repository"), + object({ + type: string("Type of the repository"), + url: string("URL of the repository"), + directory: string("Directory of the repository") + }) + ]) + ), + kunkun: KunkunExtManifest, + files: array(string("Files to include in the extension. e.g. ['dist']")) +}) +export type ExtPackageJson = InferOutput + +/** + * Extra fields for ExtPackageJson + * e.g. path to the extension + */ +export const ExtPackageJsonExtra = object({ + ...ExtPackageJson.entries, + ...{ + extPath: string(), + extFolderName: string() + } +}) + +export type ExtPackageJsonExtra = InferOutput diff --git a/packages/api/src/models/mdns.ts b/packages/api/src/models/mdns.ts new file mode 100644 index 0000000..5d0f426 --- /dev/null +++ b/packages/api/src/models/mdns.ts @@ -0,0 +1,16 @@ +import { array, number, object, optional, record, string, type InferOutput } from "valibot" + +export const MdnsServiceInfo = object({ + addresses: array(string()), + fullname: string(), + hostname: string(), + port: number(), + service_type: string(), + subType: optional(string()) +}) + +export type MdnsServiceInfo = InferOutput + +export const MdnsPeers = record(string(), MdnsServiceInfo) + +export type MdnsPeers = InferOutput diff --git a/packages/api/src/models/sql.ts b/packages/api/src/models/sql.ts new file mode 100644 index 0000000..50b5bc0 --- /dev/null +++ b/packages/api/src/models/sql.ts @@ -0,0 +1,22 @@ +import { enum_, type InferOutput } from "valibot" + +export enum SQLSortOrderEnum { + Asc = "ASC", + Desc = "DESC" +} + +export const SQLSortOrder = enum_(SQLSortOrderEnum) +export type SQLSortOrder = InferOutput + +export function convertDateToSqliteString(date: Date) { + const pad = (num: number) => num.toString().padStart(2, "0") + + const year = date.getFullYear() + const month = pad(date.getMonth() + 1) // getMonth() returns 0-11 + const day = pad(date.getDate()) + const hours = pad(date.getHours()) + const minutes = pad(date.getMinutes()) + const seconds = pad(date.getSeconds()) + + return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}` +} diff --git a/packages/api/src/models/styles.ts b/packages/api/src/models/styles.ts new file mode 100644 index 0000000..7170d7c --- /dev/null +++ b/packages/api/src/models/styles.ts @@ -0,0 +1,49 @@ +import { + hexColor, + literal, + maxValue, + minValue, + number, + object, + optional, + pipe, + string, + union, + type InferOutput +} from "valibot" + +/* -------------------------------------------------------------------------- */ +/* Color */ +/* -------------------------------------------------------------------------- */ +export const Color = pipe(string(), hexColor()) +export type Color = InferOutput +/* -------------------------------------------------------------------------- */ +/* Style */ +/* -------------------------------------------------------------------------- */ +export const CustomPosition = object({ + top: optional(number()), + right: optional(number()), + bottom: optional(number()), + left: optional(number()) +}) +export type CustomPosition = InferOutput +export type Position = "top-left" | "top-right" | "bottom-left" | "bottom-right" | CustomPosition +export const LightMode = union([literal("light"), literal("dark"), literal("auto")]) +export type LightMode = InferOutput +export const ThemeColor = union([ + literal("zinc"), + literal("slate"), + literal("stone"), + literal("gray"), + literal("neutral"), + literal("red"), + literal("rose"), + literal("orange"), + literal("green"), + literal("blue"), + literal("yellow"), + literal("violet") +]) +export type ThemeColor = InferOutput +export const Radius = pipe(number(), minValue(0), maxValue(1)) +export type Radius = InferOutput diff --git a/packages/api/src/permissions/description.ts b/packages/api/src/permissions/description.ts new file mode 100644 index 0000000..064e70b --- /dev/null +++ b/packages/api/src/permissions/description.ts @@ -0,0 +1,100 @@ +import type { AllKunkunPermission } from "./schema" + +export type PermissionDescriptions = Record + +export const permissionDescriptions: PermissionDescriptions = { + "system:volumn": "Allows access to control system volume", + "system:boot": "Allows sleep, restart, logout and shutdown, etc.", + "system:disk": "Allows access to control disk (e.g. eject)", + "system:apps": "Allows access to applications (e.g. find frontmost app, kill app)", + "system:fs": "Allows access to the clean trash, get selected files", + "system:ui": "Allows access to system UI components", + "clipboard:read-all": "Allows reading all clipboard data", + "clipboard:write-all": "Allows writing all clipboard data", + "clipboard:read-text": "Allows reading text/html/rtf from the clipboard", + "clipboard:write-text": "Allows writing text/html/rtf to the clipboard", + "clipboard:read-image": "Allows reading images from the clipboard", + "clipboard:write-image": "Allows writing images to the clipboard", + "clipboard:read-files": "Allows reading copied files from the clipboard", + "clipboard:write-files": "Allows writing files to the clipboard", + "dialog:all": "Allows access to system dialog APIs, e.g. confirm, save, open, etc.", + "notification:all": "Allows sending system notifications", + "os:all": "Allows access to all operating system information", + "shell:open": "Allows opening shell commands", + "shell:execute": "Allows executing shell commands", + "shell:spawn": "Allow spawning a new process and listen to the streaming of its output", + "shell:kill": "Allows killing processes by pid. Need this to kill the process you started.", + "shell:all": + "Grant all shell related permissions. Path scope and args regex validation is still required.", + "shell:stdin-write": "Allows writing to a command created.", + "shell:deno:execute": "Allows executing deno script", + "shell:deno:spawn": "Allow spawning a new deno process and listen to the streaming of its output", + "fetch:all": "Allows making network requests", + "system-info:all": "Allows access to all system information", + "system-info:memory": "Allows access to system memory information", + "system-info:cpu": "Allows access to system CPU information", + "system-info:os": "Allows access to operating system information", + "system-info:disk": "Allows access to disk information", + "system-info:network": "Allows access to network information", + "system-info:battery": "Allows access to battery information", + "system-info:process": "Allows access to process information", + "system-info:components": "Allows access to system components information", + "network:interface": "Allows access to network interface information", + "network:port": "Allows access to network port information", + "updownload:download": "Allows downloading files", + "updownload:upload": "Allows uploading files", + /* -------------------------------------------------------------------------- */ + /* File System */ + /* -------------------------------------------------------------------------- */ + // fs:allow-app-write-recursive + "fs:search": "Allows searching files in the file system", + "fs:read": "Allows reading files from the file system", + "fs:read-dir": + "Allows reading directories from the file system. Permissions to read files not granted, must be declared separately", + "fs:write": "Allows writing files to the file system", + "fs:exists": "Allows checking if a file exists in the file system", + "fs:stat": "Allows getting file metadata from the file system", + /* ---------------------------- File System Scope --------------------------- */ + // "fs:scope-download-recursive": "Allows reading files from the download directory and its subdirectories", + // "fs:allow-desktop-read-recursive": + // "Allows reading files from the desktop directory and its subdirectories", + // "fs:allow-desktop-write-recursive": + // "Allows writing files to the desktop directory and its subdirectories", + // "fs:allow-documents-read-recursive": + // "Allows reading files from the documents directory and its subdirectories", + // "fs:allow-documents-write-recursive": + // "Allows writing files to the documents directory and its subdirectories", + // "fs:allow-downloads-read-recursive": + // "Allows reading files from the download directory and its subdirectories", + // "fs:allow-downloads-write-recursive": + // "Allows writing files to the download directory and its subdirectories", + // "fs:scope-download-recursive": + // "Allow reading files from the download directory and its subdirectories", + // "fs:scope-desktop-recursive": + // "Allow reading files from the desktop directory and its subdirectories", + // "fs:scope-documents-recursive": + // "Allow reading files from the documents directory and its subdirectories" + "open:url": "Open URLs", + "open:file": "Open Files", + "open:folder": "Open Folders", + /* -------------------------------------------------------------------------- */ + /* Event */ + /* -------------------------------------------------------------------------- */ + "event:drag-drop": "Listen to file drop event", + "event:drag-enter": "Listen to drag enter event", + "event:drag-leave": "Listen to drag leave event", + "event:drag-over": "Listen to drag over event", + "event:window-blur": "Listen to window blur event", + "event:window-close-requested": "Listen to window close requested event", + "event:window-focus": "Listen to window focus event", + /* -------------------------------------------------------------------------- */ + /* Security */ + /* -------------------------------------------------------------------------- */ + "security:mac:reveal-security-pane": + "Reveal security privacy settings panel in Mac's System Preferences", + "security:mac:verify-fingerprint": "Verify fingerprint", + "security:mac:reset-screencapture-permission": "Reset permission", + "security:mac:request-permission": "Request security permission", + "security:mac:check-permission": "Check security permission", + "security:mac:all": "All Mac Security APIs" +} diff --git a/packages/api/src/permissions/index.ts b/packages/api/src/permissions/index.ts new file mode 100644 index 0000000..7a0856a --- /dev/null +++ b/packages/api/src/permissions/index.ts @@ -0,0 +1,3 @@ +export * from "./description" +export * from "./schema" +export * from "./permission-map" diff --git a/packages/api/src/permissions/permission-map.ts b/packages/api/src/permissions/permission-map.ts new file mode 100644 index 0000000..35e95ad --- /dev/null +++ b/packages/api/src/permissions/permission-map.ts @@ -0,0 +1,111 @@ +import type { IShellServer } from "tauri-api-adapter" +// import type { IEventServer, IFsServer, ISystemServer } from "../ui/server/server-types" +import type { IEvent, IFs, ISecurity, ISystem } from "../ui/client" +import type { + EventPermission, + KunkunFsPermission, + SecurityPermission, + ShellPermission, + SystemPermission +} from "./schema" + +/* -------------------------------------------------------------------------- */ +/* Re-export */ +/* -------------------------------------------------------------------------- */ +export { + ClipboardPermissionMap, + DialogPermissionMap, + NotificationPermissionMap, + // FsPermissionMap, + OsPermissionMap, + FetchPermissionMap, + SystemInfoPermissionMap, + // ShellPermissionMap, // we defined a custom one below + UpdownloadPermissionMap +} from "tauri-api-adapter/permissions" + +export const SecurityPermissionMap: { mac: Record } = + { + mac: { + revealSecurityPane: ["security:mac:all", "security:mac:reveal-security-pane"], + verifyFingerprint: ["security:mac:all", "security:mac:verify-fingerprint"], + requestScreenCapturePermission: ["security:mac:all", "security:mac:request-permission"], + checkScreenCapturePermission: ["security:mac:all", "security:mac:check-permission"], + resetPermission: ["security:mac:all", "security:mac:reset-screencapture-permission"] + } + } + +export const FsPermissionMap: Record = { + readDir: ["fs:read", "fs:read-dir"], + readFile: ["fs:read"], + readTextFile: ["fs:read"], + stat: ["fs:stat", "fs:read"], + lstat: ["fs:stat", "fs:read"], + exists: ["fs:exists", "fs:read"], + mkdir: ["fs:write"], + create: ["fs:write"], + copyFile: ["fs:write"], + remove: ["fs:write"], + rename: ["fs:write"], + truncate: ["fs:write"], + writeFile: ["fs:write"], + writeTextFile: ["fs:write"], + fileSearch: ["fs:search"] +} + +export const SystemPermissionMap: Record = { + openTrash: [], + emptyTrash: ["system:fs"], + shutdown: ["system:boot"], + reboot: ["system:boot"], + sleep: ["system:boot"], + toggleSystemAppearance: ["system:ui"], + showDesktop: ["system:ui"], + quitAllApps: ["system:apps"], + sleepDisplays: ["system:boot"], + setVolume: ["system:volumn"], + setVolumeTo0: ["system:volumn"], + setVolumeTo25: ["system:volumn"], + setVolumeTo50: ["system:volumn"], + setVolumeTo75: ["system:volumn"], + setVolumeTo100: ["system:volumn"], + turnVolumeUp: ["system:volumn"], + turnVolumeDown: ["system:volumn"], + toggleStageManager: ["system:ui"], + toggleBluetooth: [], + toggleHiddenFiles: ["system:ui"], + ejectAllDisks: ["system:disk"], + logoutUser: ["system:boot"], + toggleMute: ["system:volumn"], + mute: ["system:volumn"], + unmute: ["system:volumn"], + getFrontmostApp: ["system:apps"], + hideAllAppsExceptFrontmost: ["system:apps"], + getSelectedFilesInFileExplorer: ["system:fs"] +} + +export const EventPermissionMap: Record = { + onDragDrop: ["event:drag-drop"], + onDragEnter: ["event:drag-enter"], + onDragLeave: ["event:drag-leave"], + onDragOver: ["event:drag-over"], + onWindowBlur: ["event:window-blur"], + onWindowCloseRequested: ["event:window-close-requested"], + onWindowFocus: ["event:window-focus"] +} + +export const ShellPermissionMap: Record = { + execute: ["shell:all", "shell:execute"], + kill: ["shell:all", "shell:kill"], + stdinWrite: ["shell:all", "shell:stdin-write", "shell:execute"], + open: ["shell:all", "shell:open"], + rawSpawn: ["shell:all", "shell:spawn"], + executeBashScript: ["shell:all", "shell:execute"], + executePowershellScript: ["shell:all", "shell:execute"], + executeAppleScript: ["shell:all", "shell:execute"], + executePythonScript: ["shell:all", "shell:execute"], + executeZshScript: ["shell:all", "shell:execute"], + executeNodeScript: ["shell:all", "shell:execute"], + hasCommand: [], + likelyOnWindows: ["shell:all", "shell:execute"] +} diff --git a/packages/api/src/permissions/schema.ts b/packages/api/src/permissions/schema.ts new file mode 100644 index 0000000..cfb69e4 --- /dev/null +++ b/packages/api/src/permissions/schema.ts @@ -0,0 +1,168 @@ +import { + ClipboardPermissionSchema, + DialogPermissionSchema, + FetchPermissionSchema, + FsPermissionSchema, + NetworkPermissionSchema, + NotificationPermissionSchema, + OsPermissionSchema, + // ShellPermissionSchema, + SystemInfoPermissionSchema, + // AllPermissionSchema as TauriApiAdapterAllPermissionSchema, + UpdownloadPermissionSchema +} from "tauri-api-adapter/permissions" +import { array, literal, object, optional, string, union, type InferOutput } from "valibot" + +export const SystemPermissionSchema = union([ + literal("system:volumn"), + literal("system:boot"), + literal("system:disk"), + literal("system:apps"), + literal("system:fs"), + literal("system:ui") +]) +export const KunkunFsPermissionSchema = union([ + FsPermissionSchema, + literal("fs:read-dir"), + literal("fs:stat"), + literal("fs:search") +]) +export const EventPermissionSchema = union([ + literal("event:drag-drop"), + literal("event:drag-enter"), + literal("event:drag-leave"), + literal("event:drag-over"), + literal("event:window-blur"), + literal("event:window-close-requested"), + literal("event:window-focus") +]) +export const SecurityPermissionSchema = union([ + literal("security:mac:reveal-security-pane"), + literal("security:mac:verify-fingerprint"), + literal("security:mac:reset-screencapture-permission"), + literal("security:mac:request-permission"), + literal("security:mac:check-permission"), + literal("security:mac:all") +]) +export type SecurityPermission = InferOutput +export type EventPermission = InferOutput +// export const DenoRuntimePermissionSchema = union([ +// literal("deno:net"), +// literal("deno:env"), +// literal("deno:read"), +// literal("deno:write"), +// literal("deno:run"), +// literal("deno:ffi"), +// literal("deno:sys") +// ]) +// export type DenoRuntimePermission = InferOutput +export const DenoSysOptions = union([ + literal("hostname"), + literal("osRelease"), + literal("osUptime"), + literal("loadavg"), + literal("networkInterfaces"), + literal("systemMemoryInfo"), + literal("uid"), + literal("gid") +]) + +export type DenoSysOptions = InferOutput +export const DenoPermissionScopeSchema = object({ + /* ------------------------------ Deno Related ------------------------------ */ + // net: optional(array(string())), + // env: optional(array(string())), + // read: optional(array(string())), + // write: optional(array(string())), + // run: optional(array(string())), + // ffi: optional(array(string())), + // sys: optional(array(DenoSysOptions)), + net: optional(union([literal("*"), array(string())])), + env: optional(union([literal("*"), array(string())])), + read: optional(union([literal("*"), array(string())])), + write: optional(union([literal("*"), array(string())])), + run: optional(union([literal("*"), array(string())])), + ffi: optional(union([literal("*"), array(string())])), + sys: optional(union([literal("*"), array(DenoSysOptions)])) +}) +export const PermissionScopeSchema = object({ + path: optional(string()), + url: optional(string()), + cmd: optional( + object({ + program: string(), + args: array(string()) + }) + ), + ...DenoPermissionScopeSchema.entries +}) + +// export const DenoPermissionScopedSchema = object({ +// permission: DenoRuntimePermissionSchema, +// allow: optional(array(PermissionScopeSchema)), +// deny: optional(array(PermissionScopeSchema)) +// }) +// export type DenoPermissionScoped = InferOutput +export type KunkunFsPermission = InferOutput +export const FsPermissionScopedSchema = object({ + permission: KunkunFsPermissionSchema, + allow: optional(array(PermissionScopeSchema)), + deny: optional(array(PermissionScopeSchema)) +}) +export type FsPermissionScoped = InferOutput + +export const OpenPermissionSchema = union([ + literal("open:url"), + literal("open:file"), + literal("open:folder") +]) +export const OpenPermissionScopedSchema = object({ + permission: OpenPermissionSchema, + allow: optional(array(PermissionScopeSchema)), + deny: optional(array(PermissionScopeSchema)) +}) +export type OpenPermissionScoped = InferOutput + +export const ShellPermissionSchema = union([ + literal("shell:execute"), + literal("shell:deno:execute"), + literal("shell:spawn"), + literal("shell:deno:spawn"), + literal("shell:open"), + literal("shell:kill"), + literal("shell:all"), + literal("shell:stdin-write") +]) +export const ShellPermissionScopedSchema = object({ + permission: ShellPermissionSchema, + allow: optional(array(PermissionScopeSchema)), + deny: optional(array(PermissionScopeSchema)) +}) +export type ShellPermissionScoped = InferOutput +export type ShellPermission = InferOutput + +export type SystemPermission = InferOutput +export const KunkunManifestPermission = union([ + // TauriApiAdapterAllPermissionSchema, + ClipboardPermissionSchema, + EventPermissionSchema, + DialogPermissionSchema, + NotificationPermissionSchema, + // FsPermissionSchema, + OsPermissionSchema, + ShellPermissionSchema, + // StringShellPermissionSchema, // permission like exeucte and spawn must be scoped, this schema should only contain string permissions + FetchPermissionSchema, + SystemInfoPermissionSchema, + NetworkPermissionSchema, + UpdownloadPermissionSchema, + SystemPermissionSchema, + SecurityPermissionSchema + // FsScopePermissionSchema +]) +export const AllKunkunPermission = union([ + KunkunManifestPermission, + KunkunFsPermissionSchema, + OpenPermissionSchema +]) +export type AllKunkunPermission = InferOutput diff --git a/packages/api/src/runtime/deno.ts b/packages/api/src/runtime/deno.ts new file mode 100644 index 0000000..dfa39ec --- /dev/null +++ b/packages/api/src/runtime/deno.ts @@ -0,0 +1,10 @@ +import { DenoStdio, RPCChannel } from "@hk/comlink-stdio" + +// deno-lint-ignore no-explicit-any +export function expose(api: Record) { + const stdio = new DenoStdio(Deno.stdin.readable, Deno.stdout.writable) + const channel = new RPCChannel(stdio, api) + return channel +} + +export { DenoStdio, RPCChannel } from "@hk/comlink-stdio" diff --git a/packages/api/src/supabase/database.types.ts b/packages/api/src/supabase/database.types.ts new file mode 100644 index 0000000..d00a4b5 --- /dev/null +++ b/packages/api/src/supabase/database.types.ts @@ -0,0 +1,268 @@ +export type Json = string | number | boolean | null | { [key: string]: Json | undefined } | Json[] + +export type Database = { + public: { + Tables: { + events: { + Row: { + created_at: string + data: Json | null + event_type: Database["public"]["Enums"]["event_type"] + id: number + ip: string + } + Insert: { + created_at?: string + data?: Json | null + event_type: Database["public"]["Enums"]["event_type"] + id?: number + ip: string + } + Update: { + created_at?: string + data?: Json | null + event_type?: Database["public"]["Enums"]["event_type"] + id?: number + ip?: string + } + Relationships: [] + } + ext_images: { + Row: { + created_at: string + image_path: string + sha512: string + } + Insert: { + created_at?: string + image_path: string + sha512: string + } + Update: { + created_at?: string + image_path?: string + sha512?: string + } + Relationships: [] + } + ext_publish: { + Row: { + api_version: string | null + cmd_count: number + created_at: string + demo_images: string[] + downloads: number + id: number + identifier: string + manifest: Json + name: string + shasum: string + size: number + tarball_path: string + version: string + } + Insert: { + api_version?: string | null + cmd_count: number + created_at?: string + demo_images: string[] + downloads: number + id?: number + identifier: string + manifest: Json + name: string + shasum: string + size: number + tarball_path: string + version: string + } + Update: { + api_version?: string | null + cmd_count?: number + created_at?: string + demo_images?: string[] + downloads?: number + id?: number + identifier?: string + manifest?: Json + name?: string + shasum?: string + size?: number + tarball_path?: string + version?: string + } + Relationships: [ + { + foreignKeyName: "ext_publish_identifier_fkey" + columns: ["identifier"] + isOneToOne: false + referencedRelation: "extensions" + referencedColumns: ["identifier"] + } + ] + } + extensions: { + Row: { + api_version: string + created_at: string + downloads: number + icon: Json | null + identifier: string + long_description: string | null + name: string + readme: string | null + short_description: string + version: string + } + Insert: { + api_version: string + created_at?: string + downloads: number + icon?: Json | null + identifier: string + long_description?: string | null + name: string + readme?: string | null + short_description: string + version: string + } + Update: { + api_version?: string + created_at?: string + downloads?: number + icon?: Json | null + identifier?: string + long_description?: string | null + name?: string + readme?: string | null + short_description?: string + version?: string + } + Relationships: [] + } + } + Views: { + [_ in never]: never + } + Functions: { + get_aggregated_downloads: { + Args: Record + Returns: { + identifier: string + total_downloads: number + }[] + } + get_aggregated_downloads_with_details: { + Args: Record + Returns: { + identifier: string + total_downloads: number + name: string + short_description: string + }[] + } + increment_downloads: { + Args: { + t_identifier: string + t_version: string + } + Returns: number + } + } + Enums: { + event_type: "download" | "updater" | "schema" | "nightly_schema" + } + CompositeTypes: { + [_ in never]: never + } + } +} + +type PublicSchema = Database[Extract] + +export type Tables< + PublicTableNameOrOptions extends + | keyof (PublicSchema["Tables"] & PublicSchema["Views"]) + | { schema: keyof Database }, + TableName extends PublicTableNameOrOptions extends { schema: keyof Database } + ? keyof (Database[PublicTableNameOrOptions["schema"]]["Tables"] & + Database[PublicTableNameOrOptions["schema"]]["Views"]) + : never = never +> = PublicTableNameOrOptions extends { schema: keyof Database } + ? (Database[PublicTableNameOrOptions["schema"]]["Tables"] & + Database[PublicTableNameOrOptions["schema"]]["Views"])[TableName] extends { + Row: infer R + } + ? R + : never + : PublicTableNameOrOptions extends keyof (PublicSchema["Tables"] & PublicSchema["Views"]) + ? (PublicSchema["Tables"] & PublicSchema["Views"])[PublicTableNameOrOptions] extends { + Row: infer R + } + ? R + : never + : never + +export type TablesInsert< + PublicTableNameOrOptions extends keyof PublicSchema["Tables"] | { schema: keyof Database }, + TableName extends PublicTableNameOrOptions extends { schema: keyof Database } + ? keyof Database[PublicTableNameOrOptions["schema"]]["Tables"] + : never = never +> = PublicTableNameOrOptions extends { schema: keyof Database } + ? Database[PublicTableNameOrOptions["schema"]]["Tables"][TableName] extends { + Insert: infer I + } + ? I + : never + : PublicTableNameOrOptions extends keyof PublicSchema["Tables"] + ? PublicSchema["Tables"][PublicTableNameOrOptions] extends { + Insert: infer I + } + ? I + : never + : never + +export type TablesUpdate< + PublicTableNameOrOptions extends keyof PublicSchema["Tables"] | { schema: keyof Database }, + TableName extends PublicTableNameOrOptions extends { schema: keyof Database } + ? keyof Database[PublicTableNameOrOptions["schema"]]["Tables"] + : never = never +> = PublicTableNameOrOptions extends { schema: keyof Database } + ? Database[PublicTableNameOrOptions["schema"]]["Tables"][TableName] extends { + Update: infer U + } + ? U + : never + : PublicTableNameOrOptions extends keyof PublicSchema["Tables"] + ? PublicSchema["Tables"][PublicTableNameOrOptions] extends { + Update: infer U + } + ? U + : never + : never + +export type Enums< + PublicEnumNameOrOptions extends keyof PublicSchema["Enums"] | { schema: keyof Database }, + EnumName extends PublicEnumNameOrOptions extends { schema: keyof Database } + ? keyof Database[PublicEnumNameOrOptions["schema"]]["Enums"] + : never = never +> = PublicEnumNameOrOptions extends { schema: keyof Database } + ? Database[PublicEnumNameOrOptions["schema"]]["Enums"][EnumName] + : PublicEnumNameOrOptions extends keyof PublicSchema["Enums"] + ? PublicSchema["Enums"][PublicEnumNameOrOptions] + : never + +export type CompositeTypes< + PublicCompositeTypeNameOrOptions extends + | keyof PublicSchema["CompositeTypes"] + | { schema: keyof Database }, + CompositeTypeName extends PublicCompositeTypeNameOrOptions extends { + schema: keyof Database + } + ? keyof Database[PublicCompositeTypeNameOrOptions["schema"]]["CompositeTypes"] + : never = never +> = PublicCompositeTypeNameOrOptions extends { schema: keyof Database } + ? Database[PublicCompositeTypeNameOrOptions["schema"]]["CompositeTypes"][CompositeTypeName] + : PublicCompositeTypeNameOrOptions extends keyof PublicSchema["CompositeTypes"] + ? PublicSchema["CompositeTypes"][PublicCompositeTypeNameOrOptions] + : never diff --git a/packages/api/src/supabase/index.ts b/packages/api/src/supabase/index.ts new file mode 100644 index 0000000..c2b5335 --- /dev/null +++ b/packages/api/src/supabase/index.ts @@ -0,0 +1,2 @@ +export * from "./model" +export * from "./database.types" diff --git a/packages/api/src/supabase/model.ts b/packages/api/src/supabase/model.ts new file mode 100644 index 0000000..3c85e0b --- /dev/null +++ b/packages/api/src/supabase/model.ts @@ -0,0 +1,19 @@ +import { Icon } from "@kksh/api/models" +import * as v from "valibot" + +/*** + * Correspond to `extensions` table in supabase + */ +export const SBExt = v.object({ + identifier: v.string(), + name: v.string(), + created_at: v.string(), + downloads: v.number(), + short_description: v.string(), + long_description: v.string(), + version: v.string(), + api_version: v.optional(v.string()), + icon: Icon +}) + +export type SBExt = v.InferOutput diff --git a/packages/api/src/ui/README.md b/packages/api/src/ui/README.md new file mode 100644 index 0000000..e69de29 diff --git a/packages/api/src/ui/api/deno.ts b/packages/api/src/ui/api/deno.ts new file mode 100644 index 0000000..bf8e0af --- /dev/null +++ b/packages/api/src/ui/api/deno.ts @@ -0,0 +1,2 @@ +import type { DenoSysOptions } from "../../permissions/schema" +import type { DenoRunConfig } from "../client" diff --git a/packages/api/src/ui/api/event.ts b/packages/api/src/ui/api/event.ts new file mode 100644 index 0000000..e365f04 --- /dev/null +++ b/packages/api/src/ui/api/event.ts @@ -0,0 +1,17 @@ +import { proxy, type Remote } from "@huakunshen/comlink" +import type { DragDropPayload, DragEnterPayload, DragOverPayload, IEvent } from "../client" + +// event API listens for events, event callback functions are proxied with comlink, thus I have to provide this constructor function +export function constructEventAPI(api: Remote): IEvent { + return { + onDragDrop: (callback: (payload: DragDropPayload) => void) => api.onDragDrop(proxy(callback)), + onDragEnter: (callback: (payload: DragEnterPayload) => void) => + api.onDragEnter(proxy(callback)), + onDragLeave: (callback: () => void) => api.onDragLeave(proxy(callback)), + onDragOver: (callback: (payload: DragOverPayload) => void) => api.onDragOver(proxy(callback)), + onWindowBlur: (callback: () => void) => api.onWindowBlur(proxy(callback)), + onWindowCloseRequested: (callback: () => void) => api.onWindowCloseRequested(proxy(callback)), + onWindowFocus: (callback: () => void) => api.onWindowFocus(proxy(callback)) + // onWindowThemeChanged: api.onWindowThemeChanged + } +} diff --git a/packages/api/src/ui/api/iframe-ui.ts b/packages/api/src/ui/api/iframe-ui.ts new file mode 100644 index 0000000..b285393 --- /dev/null +++ b/packages/api/src/ui/api/iframe-ui.ts @@ -0,0 +1,93 @@ +import type { Remote } from "@huakunshen/comlink" +import type { IOs } from "tauri-api-adapter/client" +import { type IUiIframe } from "../client" + +export const KK_DRAG_REGION_ATTR = "data-kunkun-drag-region" + +export function constructIframeUiAPI(api: { + iframeUi: Remote + os: Remote +}): IUiIframe { + return { + goBack: api.iframeUi.goBack, + hideBackButton: api.iframeUi.hideBackButton, + hideMoveButton: api.iframeUi.hideMoveButton, + hideRefreshButton: api.iframeUi.hideRefreshButton, + showBackButton: api.iframeUi.showBackButton, + showMoveButton: api.iframeUi.showMoveButton, + showRefreshButton: api.iframeUi.showRefreshButton, + getTheme: api.iframeUi.getTheme, + reloadPage: api.iframeUi.reloadPage, + toggleMaximize: api.iframeUi.toggleMaximize, + startDragging: api.iframeUi.startDragging, + internalToggleMaximize: api.iframeUi.internalToggleMaximize, + setTransparentWindowBackground: api.iframeUi.setTransparentWindowBackground, + /** + * https://github.com/tauri-apps/tauri/blob/e1776946ad034d7a6e005834a754773671d9f7ef/core/tauri/src/window/scripts/drag.js#L13 + */ + registerDragRegion: async (): Promise => { + const osName = await api.os.platform() + let x = 0 + let y = 0 + + document.addEventListener("mousedown", (e) => { + const target = e.target as HTMLElement + if ( + // element has the magic data attribute + (target.classList.contains(KK_DRAG_REGION_ATTR) || + target.hasAttribute(KK_DRAG_REGION_ATTR)) && + // and was left mouse button + e.button === 0 && + // and was normal click to drag or double click to maximize + (e.detail === 1 || e.detail === 2) + ) { + // macOS maximization happens on `mouseup`, + // so we save needed state and early return + if (osName === "macos" && e.detail == 2) { + x = e.clientX + y = e.clientY + return + } + + // prevents text cursor + e.preventDefault() + + // fix #2549: double click on drag region edge causes content to maximize without window sizing change + // https://github.com/tauri-apps/tauri/issues/2549#issuecomment-1250036908 + e.stopImmediatePropagation() + + // start dragging if the element has a `tauri-drag-region` data attribute and maximize on double-clicking it + // const cmd = e.detail === 2 ? "internal_toggle_maximize" : "start_dragging" + if (e.detail === 2) { + api.iframeUi.internalToggleMaximize() + } else { + api.iframeUi.startDragging() + } + // window.__TAURI_INTERNALS__.invoke("plugin:window|" + cmd) + } + }) + + if (osName === "macos") { + document.addEventListener("mouseup", (e) => { + const target = e.target as HTMLElement + if ( + // element has the magic data attribute + (target.classList.contains(KK_DRAG_REGION_ATTR) || + target.hasAttribute(KK_DRAG_REGION_ATTR)) && + // target.hasAttribute(KK_DRAG_REGION_ATTR) && + // and was left mouse button + e.button === 0 && + // and was double click + e.detail === 2 && + // and the cursor hasn't moved from initial mousedown + e.clientX === x && + e.clientY === y + ) { + // window.__TAURI_INTERNALS__.invoke("plugin:window|internal_toggle_maximize") + api.iframeUi.internalToggleMaximize() + } + }) + } + } + } +} diff --git a/packages/api/src/ui/api/path.ts b/packages/api/src/ui/api/path.ts new file mode 100644 index 0000000..1e9081d --- /dev/null +++ b/packages/api/src/ui/api/path.ts @@ -0,0 +1,45 @@ +import type { Remote } from "@huakunshen/comlink" +import { BaseDirectory } from "@tauri-apps/api/path" +import type { IPath } from "../client" + +export function constructPathAPI(api: Remote): IPath { + return { + BaseDirectory: BaseDirectory, + appCacheDir: api.appCacheDir, + appConfigDir: api.appConfigDir, + appDataDir: api.appDataDir, + appLocalDataDir: api.appLocalDataDir, + appLogDir: api.appLogDir, + audioDir: api.audioDir, + basename: api.basename, + cacheDir: api.cacheDir, + configDir: api.configDir, + dataDir: api.dataDir, + delimiter: api.delimiter, + desktopDir: api.desktopDir, + dirname: api.dirname, + documentDir: api.documentDir, + downloadDir: api.downloadDir, + executableDir: api.executableDir, + extname: api.extname, + fontDir: api.fontDir, + homeDir: api.homeDir, + isAbsolute: api.isAbsolute, + join: api.join, + localDataDir: api.localDataDir, + normalize: api.normalize, + pictureDir: api.pictureDir, + publicDir: api.publicDir, + resolve: api.resolve, + resolveResource: api.resolveResource, + resourceDir: api.resourceDir, + runtimeDir: api.runtimeDir, + sep: api.sep, + tempDir: api.tempDir, + templateDir: api.templateDir, + videoDir: api.videoDir, + /* --------------------------- Custom Directories --------------------------- */ + extensionDir: api.extensionDir, + extensionSupportDir: api.extensionSupportDir + } +} diff --git a/packages/api/src/ui/api/shell.ts b/packages/api/src/ui/api/shell.ts new file mode 100644 index 0000000..69da7c4 --- /dev/null +++ b/packages/api/src/ui/api/shell.ts @@ -0,0 +1,405 @@ +import { type Buffer } from "node:buffer" +import { RPCChannel, type StdioInterface } from "@hk/comlink-stdio/browser" +import { proxy as comlinkProxy, type Remote } from "@huakunshen/comlink" +import { Channel, invoke } from "@tauri-apps/api/core" +import { constructShellAPI as constructShellAPI1 } from "tauri-api-adapter/client" +import { + // Child, + EventEmitter, + hasCommand, + likelyOnWindows, + open as shellxOpen, + // EventEmitter, + // Command as ShellxCommand, + type ChildProcess, + type CommandEvent, + type CommandEvents, + type InternalSpawnOptions, + type IOPayload, + type OutputEvents, + type SpawnOptions +} from "tauri-plugin-shellx-api" +import { type DenoRunConfig } from "../client.ts" +import type { IShellServer } from "../server/server-types.ts" + +export class Child { + /** The child process `pid`. */ + pid: number + api: Remote + constructor(pid: number, api: Remote) { + this.pid = pid + this.api = api + } + + /** + * Writes `data` to the `stdin`. + * + * @param data The message to write, either a string or a byte array. + * @example + * ```typescript + * import { Command } from '@tauri-apps/plugin-shell'; + * const command = Command.create('node'); + * const child = await command.spawn(); + * await child.write('message'); + * await child.write([0, 1, 2, 3, 4, 5]); + * ``` + * + * @returns A promise indicating the success or failure of the operation. + * + * @since 2.0.0 + */ + async write(data: IOPayload | number[]): Promise { + // await invoke('plugin:shellx|stdin_write', { + // pid: this.pid, + // buffer: data + // }) + this.api.stdinWrite(data.toString(), this.pid) + } + + /** + * Kills the child process. + * + * @returns A promise indicating the success or failure of the operation. + * + * @since 2.0.0 + */ + async kill(): Promise { + this.api.kill(this.pid) + // await invoke("plugin:shellx|kill", { + // cmd: "killChild", + // pid: this.pid + // }) + } +} + +class BaseShellCommand extends EventEmitter { + /** @ignore Program to execute. */ + readonly program: string + /** @ignore Program arguments */ + readonly args: string[] + /** @ignore Spawn options. */ + readonly options: SpawnOptions + /** Event emitter for the `stdout`. Emits the `data` event. */ + readonly stdout = new EventEmitter>() + /** Event emitter for the `stderr`. Emits the `data` event. */ + readonly stderr = new EventEmitter>() + + /** + * @ignore + * Creates a new `Command` instance. + * + * @param program The program name to execute. + * It must be configured on `tauri.conf.json > plugins > shell > scope`. + * @param args Program arguments. + * @param options Spawn options. + */ + constructor(program: string, args: string | string[] = [], options?: SpawnOptions) { + super() + this.program = program + this.args = typeof args === "string" ? [args] : args + this.options = options ?? {} + } +} + +class Command extends BaseShellCommand { + api: Remote + + constructor( + program: string, + args: string | string[] = [], + api: Remote, + options?: SpawnOptions + ) { + super(program, args, options) + this.api = api + } + + async spawn(): Promise { + const args = this.args + + if (typeof args === "object") { + Object.freeze(args) + } + + return this.api + .rawSpawn( + this.program, + args, + this.options, + comlinkProxy((evt) => { + switch (evt.event) { + case "Error": + this.emit("error", evt.payload) + break + case "Terminated": + this.emit("close", evt.payload) + break + case "Stdout": + this.stdout.emit("data", evt.payload as O) + break + case "Stderr": + this.stderr.emit("data", evt.payload as O) + break + } + }) + ) + .then(async (pid) => { + await this.api.recordSpawnedProcess(pid) // report spawned process to main process for process auto cleanup + return new Child(pid, this.api) + }) + } + + async execute(): Promise> { + const args = this.args + if (typeof args === "object") { + Object.freeze(args) + } + return this.api.execute(this.program, this.args, this.options) as Promise> + } +} + +class DenoCommand extends BaseShellCommand { + config: DenoRunConfig + scriptPath: string + api: Remote + + constructor( + scriptPath: string, + args: string[], + config: DenoRunConfig, + api: Remote + ) { + super("deno", args) + this.config = config + this.scriptPath = scriptPath + this.api = api + } + + execute(): Promise> { + return this.api.denoExecute(this.scriptPath, this.config, this.args) as Promise> + } + + spawn(): Promise { + return this.api + .denoRawSpawn( + this.scriptPath, + this.config, + this.args, + comlinkProxy((evt) => { + switch (evt.event) { + case "Error": + this.emit("error", evt.payload) + break + case "Terminated": + this.emit("close", evt.payload) + break + case "Stdout": + this.stdout.emit("data", evt.payload as O) + break + case "Stderr": + this.stderr.emit("data", evt.payload as O) + break + } + }) + ) + .then(async (pid) => { + await this.api.recordSpawnedProcess(pid) + return new Child(pid, this.api) + }) + } +} + +export type IShell = { + // Command: typeof Command + // DenoCommand: typeof DenoCommand + createCommand: ( + program: string, + args: string | string[], + options?: SpawnOptions + ) => Command + createDenoCommand: ( + scriptPath: string, + args: string[], + config: Partial & SpawnOptions + ) => DenoCommand + open: typeof shellxOpen + makeBashScript: (script: string) => Command + makePowershellScript: (script: string) => Command + makeAppleScript: (script: string) => Command + makePythonScript: (script: string) => Command + makeZshScript: (script: string) => Command + makeNodeScript: (script: string) => Command + executeBashScript: (script: string) => Promise> + executePowershellScript: (script: string) => Promise> + executeAppleScript: (script: string) => Promise> + executePythonScript: (script: string) => Promise> + executeZshScript: (script: string) => Promise> + executeNodeScript: (script: string) => Promise> + hasCommand: typeof hasCommand + likelyOnWindows: typeof likelyOnWindows + Child: typeof Child + TauriShellStdio: typeof TauriShellStdio + createDenoRpcChannel( + scriptPath: string, + args: string[], + config: Partial & SpawnOptions, + localAPIImplementation: LocalAPI + ): Promise<{ + rpcChannel: RPCChannel + process: Child + }> + RPCChannel: typeof RPCChannel +} + +export class TauriShellStdio implements StdioInterface { + constructor( + private readStream: EventEmitter>, // stdout of child process + private childProcess: Child + ) {} + + read(): Promise { + return new Promise((resolve, reject) => { + this.readStream.on("data", (chunk) => { + resolve(chunk) + }) + }) + } + async write(data: string): Promise { + return this.childProcess.write(data + "\n") + } +} + +export function constructShellAPI(api: Remote): IShell { + // const originalAPI = constructShellAPI1(api) + function createCommand(program: string, args: string | string[] = [], options?: SpawnOptions) { + return new Command(program, args, api, options) + } + + function createDenoCommand( + scriptPath: string, + args: string[], + config: Partial & SpawnOptions + ) { + return new DenoCommand(scriptPath, args, config, api) + } + + async function createDenoRpcChannel( + scriptPath: string, + args: string[], + config: Partial & SpawnOptions, + localAPIImplementation: LocalAPI + ) { + const denoCmd = createDenoCommand(scriptPath, args, config) + const denoProcess = await denoCmd.spawn() + const stdio = new TauriShellStdio(denoCmd.stdout, denoProcess) + const stdioRPC = new RPCChannel(stdio, localAPIImplementation) + return { + rpcChannel: stdioRPC, + process: denoProcess + } + } + + function makeBashScript(script: string): Command { + return createCommand("bash", ["-c", script]) + } + + function makePowershellScript(script: string): Command { + return createCommand("powershell", ["-Command", script]) + } + + function makeAppleScript(script: string): Command { + return createCommand("osascript", ["-e", script]) + } + + function makePythonScript(script: string): Command { + return createCommand("python", ["-c", script]) + } + + function makeZshScript(script: string): Command { + return createCommand("zsh", ["-c", script]) + } + + function makeNodeScript(script: string): Command { + return createCommand("node", ["-e", script]) + } + + async function executeBashScript(script: string): Promise> { + return makeBashScript(script).execute() + } + + async function executePowershellScript(script: string): Promise> { + return makePowershellScript(script).execute() + } + + async function executeAppleScript(script: string): Promise> { + return makeAppleScript(script).execute() + } + + async function executePythonScript(script: string): Promise> { + return makePythonScript(script).execute() + } + + async function executeZshScript(script: string): Promise> { + return makeZshScript(script).execute() + } + + async function executeNodeScript(script: string): Promise> { + return makeNodeScript(script).execute() + } + + /** + * Run powershell.exe -Command 'echo $env:OS' to determine if the current platform is likely to be Windows. + * Not 100% accurate. If a Mac or Linux somehow has powershell.exe in PATH, this will return true. + * @returns Whether the current platform is likely to be Windows. + */ + function likelyOnWindows(): Promise { + return createCommand("powershell.exe", ["-Command", "echo $env:OS"]) + .execute() + .then((out) => out.code === 0 && out.stdout.toLowerCase().includes("windows")) + .catch(() => false) + } + + /** + * Determine if a command is available with `which` or `where` command. + * Support Windows, Mac, Linux + * @param command + * @returns + */ + async function hasCommand(command: string): Promise { + const targetCmd = command.trim().split(" ")[0] + if (!targetCmd) { + return false + } + const isOnWindows = await likelyOnWindows() + const whereCmd = isOnWindows ? "where" : "which" + const cmd = createCommand(whereCmd, [targetCmd]) + const out = await cmd.execute() + return out.code === 0 + } + + return { + open: api.open, + makeBashScript, + makePowershellScript, + makeAppleScript, + makePythonScript, + makeZshScript, + makeNodeScript, + executeBashScript, + executePowershellScript, + executeAppleScript, + executePythonScript, + executeZshScript, + executeNodeScript, + hasCommand, + likelyOnWindows, + createCommand, + createDenoCommand, + Child, + TauriShellStdio, + createDenoRpcChannel, + RPCChannel + } +} + +export { RPCChannel } from "@hk/comlink-stdio/browser" diff --git a/packages/api/src/ui/api/toast.ts b/packages/api/src/ui/api/toast.ts new file mode 100644 index 0000000..5102c9e --- /dev/null +++ b/packages/api/src/ui/api/toast.ts @@ -0,0 +1,17 @@ +import { proxy as comlinkProxy, type Remote } from "@huakunshen/comlink" +import type { GeneralToastParams, IToast } from "../client" + +export function constructToastAPI(api: Remote) { + return { + message: (message: string, options?: GeneralToastParams, action?: () => void) => + api.message(message, options, action ? comlinkProxy(action) : undefined), + info: (message: string, options?: GeneralToastParams, action?: () => void) => + api.info(message, options, action ? comlinkProxy(action) : undefined), + success: (message: string, options?: GeneralToastParams, action?: () => void) => + api.success(message, options, action ? comlinkProxy(action) : undefined), + warning: (message: string, options?: GeneralToastParams, action?: () => void) => + api.warning(message, options, action ? comlinkProxy(action) : undefined), + error: (message: string, options?: GeneralToastParams, action?: () => void) => + api.error(message, options, action ? comlinkProxy(action) : undefined) + } +} diff --git a/packages/api/src/ui/api/worker-ui.ts b/packages/api/src/ui/api/worker-ui.ts new file mode 100644 index 0000000..c550cc0 --- /dev/null +++ b/packages/api/src/ui/api/worker-ui.ts @@ -0,0 +1,12 @@ +// import { proxy, type Remote } from "@huakunshen/comlink" +// import type { IUiWorker } from "../client" + +// export function constructWorkerUiAPI(api: Remote): IUiWorker { +// return { +// goBack: api.goBack, +// render: api.render, +// setScrollLoading: api.setScrollLoading, +// setSearchTerm: api.setSearchTerm, +// setSearchBarPlaceholder: api.setSearchBarPlaceholder, +// } +// } diff --git a/packages/api/src/ui/client.ts b/packages/api/src/ui/client.ts new file mode 100644 index 0000000..5b66568 --- /dev/null +++ b/packages/api/src/ui/client.ts @@ -0,0 +1,302 @@ +import type { + copyFile, + create, + exists, + lstat, + mkdir, + readDir, + readFile, + readTextFile, + remove, + rename, + stat, + truncate, + writeFile, + writeTextFile +} from "@tauri-apps/plugin-fs" +import type { IShell as IShell1, IPath as ITauriPath } from "tauri-api-adapter" +import type { + Child, + ChildProcess, + CommandEvents, + hasCommand, + InternalSpawnOptions, + IOPayload, + likelyOnWindows, + OutputEvents, + SpawnOptions +} from "tauri-plugin-shellx-api" +import { EventEmitter, open as shellxOpen } from "tauri-plugin-shellx-api" +import * as v from "valibot" +import { type JarvisExtDB } from "../commands/db" +import type { fileSearch } from "../commands/fileSearch" +import { type AppInfo } from "../models/apps" +import type { LightMode, Position, Radius, ThemeColor } from "../models/styles" +import type { DenoSysOptions } from "../permissions/schema" +import { type IComponent } from "./worker/components/interfaces" +import type { Markdown } from "./worker/components/markdown" +import * as FormSchema from "./worker/schema/form" +import * as ListSchema from "./worker/schema/list" + +type PromiseWrap any> = ( + ...args: Parameters +) => Promise> + +export type IPath = ITauriPath & { + extensionDir: () => Promise + extensionSupportDir: () => Promise +} + +export interface IPlist { + // build: PromiseWrap + parse: (plistContent: string) => Promise +} + +export interface IUtils { + plist: IPlist +} + +export interface ISystem { + openTrash(): Promise + emptyTrash(): Promise + shutdown(): Promise + reboot(): Promise + sleep(): Promise + toggleSystemAppearance(): Promise + showDesktop(): Promise + quitAllApps(): Promise + sleepDisplays(): Promise + setVolume(percentage: number): Promise + setVolumeTo0(): Promise + setVolumeTo25(): Promise + setVolumeTo50(): Promise + setVolumeTo75(): Promise + setVolumeTo100(): Promise + turnVolumeUp(): Promise + turnVolumeDown(): Promise + toggleStageManager(): Promise + toggleBluetooth(): Promise + toggleHiddenFiles(): Promise + ejectAllDisks(): Promise + logoutUser(): Promise + toggleMute(): Promise + mute(): Promise + unmute(): Promise + getFrontmostApp(): Promise + hideAllAppsExceptFrontmost(): Promise + getSelectedFilesInFileExplorer(): Promise +} + +export type GeneralToastParams = { + description?: string + duration?: number + closeButton?: boolean + position?: + | "top-left" + | "top-right" + | "bottom-left" + | "bottom-right" + | "top-center" + | "bottom-center" + actionLabel?: string +} + +export type GeneralToast = ( + message: string, + options?: GeneralToastParams, + action?: () => void +) => Promise + +export interface IToast { + message: GeneralToast + info: GeneralToast + success: GeneralToast + warning: GeneralToast + error: GeneralToast +} + +export interface IUiWorker { + render: (view: IComponent) => Promise + goBack: () => Promise + showLoadingBar: (loading: boolean) => Promise + setScrollLoading: (loading: boolean) => Promise + setSearchTerm: (term: string) => Promise + setSearchBarPlaceholder: (placeholder: string) => Promise + setProgressBar: (progress: number | null) => Promise +} + +export interface IUiIframe { + // goHome: () => Promise + goBack: () => Promise + hideBackButton: () => Promise + hideMoveButton: () => Promise + hideRefreshButton: () => Promise + /** + * position can be "top-left" | "top-right" | "bottom-left" | "bottom-right" | CustomPosition + * `CustomPosition` is an object with optional `top`, `right`, `bottom`, `left` properties + * Each property is a number, with `rem` unit, and will be applied to css `top`, `right`, `bottom`, `left` properties + * @param position "top-left" | "top-right" | "bottom-left" | "bottom-right" | CustomPosition + * @example + * ```ts + * ui.showBackButton({ top: 2, left: 2 }) + * ui.showBackButton('top-right') + * ``` + * @returns + */ + showBackButton: (position?: Position) => Promise + /** + * position can be "top-left" | "top-right" | "bottom-left" | "bottom-right" | CustomPosition + * `CustomPosition` is an object with optional `top`, `right`, `bottom`, `left` properties + * Each property is a number, with `rem` unit, and will be applied to css `top`, `right`, `bottom`, `left` properties + * @param position "top-left" | "top-right" | "bottom-left" | "bottom-right" | CustomPosition + * @example + * ```ts + * ui.showBackButton({ top: 2, left: 2 }) + * ui.showBackButton('top-right') + * ``` + * @returns + */ + showMoveButton: (position?: Position) => Promise + showRefreshButton: (position?: Position) => Promise + getTheme: () => Promise<{ theme: ThemeColor; radius: Radius; lightMode: LightMode }> + reloadPage: () => Promise + startDragging: () => Promise + toggleMaximize: () => Promise + internalToggleMaximize: () => Promise + setTransparentWindowBackground: (transparent: boolean) => Promise + registerDragRegion: () => Promise +} + +export interface IDb { + add: typeof JarvisExtDB.prototype.add + delete: typeof JarvisExtDB.prototype.delete + search: typeof JarvisExtDB.prototype.search + retrieveAll: typeof JarvisExtDB.prototype.retrieveAll + retrieveAllByType: typeof JarvisExtDB.prototype.retrieveAllByType + deleteAll: typeof JarvisExtDB.prototype.deleteAll + update: typeof JarvisExtDB.prototype.update +} + +export interface IFs { + readDir: typeof readDir + readFile: typeof readFile + readTextFile: typeof readTextFile + stat: typeof stat + lstat: typeof lstat + exists: typeof exists + mkdir: typeof mkdir + create: typeof create + copyFile: typeof copyFile + remove: typeof remove + rename: typeof rename + truncate: typeof truncate + writeFile: typeof writeFile + writeTextFile: typeof writeTextFile + fileSearch: typeof fileSearch +} + +export interface IOpen { + url: (url: string) => Promise + file: (path: string) => Promise + folder: (path: string) => Promise +} + +/* -------------------------------------------------------------------------- */ +/* Event API */ +/* -------------------------------------------------------------------------- */ +export type DragDropPayload = { + paths: string[] + position: { x: number; y: number } +} +export type DragEnterPayload = DragDropPayload +export type DragOverPayload = { + position: { x: number; y: number } +} + +export interface IEvent { + /** + * Get files dropped on the window + */ + onDragDrop: (callback: (payload: DragDropPayload) => void) => void + /** + * Listen to drag enter event, when mouse drag enters the window + */ + onDragEnter: (callback: (payload: DragEnterPayload) => void) => void + /** + * Listen to drag leave event, when mouse drag leaves the window + */ + onDragLeave: (callback: () => void) => void + /** + * Get the position of the dragged item + */ + onDragOver: (callback: (payload: DragOverPayload) => void) => void + /** + * Listen to window blur (defocus) event + */ + onWindowBlur: (callback: () => void) => void + /** + * Listen to window close request event + */ + onWindowCloseRequested: (callback: () => void) => void + /** + * Listen to window on focus event + */ + onWindowFocus: (callback: () => void) => void +} + +/** + * https://docs.deno.com/runtime/fundamentals/security/ + */ +export interface DenoRunConfig { + allowNet?: string[] + allowAllNet?: boolean + allowRead?: string[] + allowAllRead?: boolean + allowWrite?: string[] + allowAllWrite?: boolean + allowRun?: string[] + allowAllRun?: boolean + allowEnv?: string[] + allowAllEnv?: boolean + allowFfi?: string[] + allowAllFfi?: boolean + allowSys?: DenoSysOptions[] + allowAllSys?: boolean + denyNet?: string[] + denyAllNet?: boolean + denyRead?: string[] + denyAllRead?: boolean + denyWrite?: string[] + denyAllWrite?: boolean + denyRun?: string[] + denyAllRun?: boolean + denyEnv?: string[] + denyAllEnv?: boolean + denyFfi?: string[] + denyAllFfi?: boolean + denySys?: DenoSysOptions[] + denyAllSys?: boolean +} + +export interface IApp { + language: () => Promise<"en" | "zh"> +} + +export const MacSecurityOptions = v.union([ + v.literal("ScreenCapture"), + v.literal("Camera"), + v.literal("Microphone"), + v.literal("Accessibility"), + v.literal("AllFiles") +]) +export type MacSecurityOptions = v.InferOutput + +export interface ISecurity { + mac: { + revealSecurityPane: (privacyOption?: MacSecurityOptions) => Promise + resetPermission: (privacyOption: MacSecurityOptions) => Promise + verifyFingerprint: () => Promise + requestScreenCapturePermission: () => Promise + checkScreenCapturePermission: () => Promise + } +} diff --git a/packages/api/src/ui/iframe/index.ts b/packages/api/src/ui/iframe/index.ts new file mode 100644 index 0000000..965c7f0 --- /dev/null +++ b/packages/api/src/ui/iframe/index.ts @@ -0,0 +1,96 @@ +import { windowEndpoint, wrap, type Remote } from "@huakunshen/comlink" +import type { + IClipboard, + IDialog, + // IEventInternal, + IFetchInternal, + // IFs, + ILogger, + INetwork, + INotification, + IOs, + // IPath, + IShellInternal, + ISystemInfo, + IUpdownload +} from "tauri-api-adapter" +import { + constructFetchAPI, + // constructPathAPI, + constructUpdownloadAPI +} from "tauri-api-adapter/client" +import { constructEventAPI } from "../api/event" +import { constructIframeUiAPI } from "../api/iframe-ui" +import { constructPathAPI } from "../api/path" +import { constructShellAPI } from "../api/shell" +import type { + IApp, + IDb, + IEvent, + IFs, + IOpen, + IPath, + ISecurity, + ISystem, + IToast, + IUiIframe, + IUtils +} from "../client" +import type { IShellServer } from "../server/server-types" + +export { type IUiIframe } from "../client" +export { expose, wrap } from "@huakunshen/comlink" +// export { type IDbServer } from "../server/db" +export { type IUiIframeServer2, type IUiIframeServer1 } from "../server/server-types" + +/** + * For the APIs annotated with "inherit from tauri-api-adapter", they inherit the client API completely from tauri-api-adapter + * There may be server API changes for them, but the client API can be inherited + */ +type API = { + db: Remote // for kunkun + system: Remote // for kunkun + open: Remote // for kunkun + clipboard: Remote // inherit from tauri-api-adapter + dialog: Remote // inherit from tauri-api-adapter + fetch: Remote // inherit from tauri-api-adapter + event: Remote // for kunkun, override tauri-api-adapter's event API, expose only specified event, disallow, emit and listen + fs: Remote // customized for kunkun, add file search API on top of tauri-api-adapter's fs API + log: Remote // inherit from tauri-api-adapter + notification: Remote // inherit from tauri-api-adapter + toast: Remote // for kunkun + os: Remote // inherit from tauri-api-adapter + path: Remote // inherit from tauri-api-adapter + shell: Remote // inherit from tauri-api-adapter + updownload: IUpdownload // inherit from tauri-api-adapter + sysInfo: Remote // inherit from tauri-api-adapter + network: Remote // inherit from tauri-api-adapter + iframeUi: Remote // for kunkun + utils: IUtils // for kunkun + security: ISecurity // for kunkun + app: IApp +} +const _api = wrap(windowEndpoint(globalThis.parent)) as unknown as API +export const event = constructEventAPI(_api.event) // this is different from event api from tauri-api-adapter +export const fetch = constructFetchAPI(_api.fetch) +export const path = constructPathAPI(_api.path) +export const shell = constructShellAPI(_api.shell) +export const updownload = constructUpdownloadAPI(_api.updownload) +export const ui = constructIframeUiAPI(_api) +export const { + db, + os, + clipboard, + dialog, + fs, + log, + notification, + sysInfo, + network, + system, + toast, + utils, + open, + app +} = _api +export { Child, RPCChannel } from "../api/shell" diff --git a/packages/api/src/ui/index.ts b/packages/api/src/ui/index.ts new file mode 100644 index 0000000..c22ecee --- /dev/null +++ b/packages/api/src/ui/index.ts @@ -0,0 +1,42 @@ +/* -------------------------------------------------------------------------- */ +/* API */ +/* -------------------------------------------------------------------------- */ +// export { +// clipboard, +// dialog, +// // event, +// network, +// fs, +// notification, +// os, +// shell, +// sysInfo, +// path, +// log, +// updownload, +// fetch +// } from "tauri-api-adapter" +export { constructJarvisServerAPIWithPermissions } from "./server" +// export { type IUiWorkerServer, type IUiIframeServer } from "./server/ui" +export * from "./client" // all client types +export { expose, wrap } from "@huakunshen/comlink" +export { getWorkerApiClient, exposeApiToWorker, exposeApiToWindow } from "tauri-api-adapter" + +/* -------------------------------------------------------------------------- */ +/* API Interfaces */ +/* -------------------------------------------------------------------------- */ +export type { + IClipboard, + IDialog, + ILogger, + INetwork, + INotification, + IOs, + IPath, + // IShell, + ISystemInfo, + IUpdownload, + IFetch +} from "tauri-api-adapter" +export type { ISystem, IToast, IUiWorker, IUiIframe, IDb, IFs, IOpen, IEvent } from "../ui/client" +export type { IShell } from "./api/shell" diff --git a/packages/api/src/ui/server/__tests__/fs.test.ts b/packages/api/src/ui/server/__tests__/fs.test.ts new file mode 100644 index 0000000..366663d --- /dev/null +++ b/packages/api/src/ui/server/__tests__/fs.test.ts @@ -0,0 +1,9 @@ +import { describe, expect, test } from "bun:test" +import { minimatch } from "minimatch" +import { translateScopeToPath } from "../../../utils/path" + +test("minimatch", () => { + // console.log(minimatch("/desktop/newbi/bar.foo", "/desktop/**/*.foo")) + // console.log("$DESKTOP/newbi/bar.foo".split("/")) + // find the first slash of "$DESKTOP/newbi/bar.foo" +}) diff --git a/packages/api/src/ui/server/__tests__/shell.test.ts b/packages/api/src/ui/server/__tests__/shell.test.ts new file mode 100644 index 0000000..24cba6e --- /dev/null +++ b/packages/api/src/ui/server/__tests__/shell.test.ts @@ -0,0 +1,36 @@ +import { mockIPC } from "@tauri-apps/api/mocks" +import { describe, expect, test } from "bun:test" +import { translateDenoCommand } from "../deno" + +// can't run this because it relies on Tauri API, I can't run it without Tauri app env, may need to mock the API +// test("translateDenoCommand", async () => { +// // mockIPC((cmd, args) => { +// // // simulated rust command called "add" that just adds two numbers +// // console.log("cmd and args", cmd, args); + +// // // if (cmd === "add") { +// // // return (args.a as number) + (args.b as number) +// // // } +// // }) +// const cmdOptions = await translateDenoCommand( +// "$EXTENSION/src/test.ts", +// { +// allowAllEnv: false, +// allowEnv: ["PATH"], +// allowAllNet: true, +// allowNet: [], +// denyAllRead: true +// }, +// [], +// "/extensions/ext" +// ) + +// expect(cmdOptions.args).toEqual([ +// "run", +// "--allow-env=PATH", +// "--allow-net", +// "--deny-read", +// "/extensions/ext/src/test.ts" +// ]) +// console.log(cmdOptions) +// }) diff --git a/packages/api/src/ui/server/db.ts b/packages/api/src/ui/server/db.ts new file mode 100644 index 0000000..ce65bcc --- /dev/null +++ b/packages/api/src/ui/server/db.ts @@ -0,0 +1,19 @@ +/** + * The server-side API for the database will not be implemented in this file/package + * It will be constructed with JarvisExtDB in the main thread and exposed to the extension + * We don't know extension ID here, so we can't construct the API here + */ +import type { JarvisExtDB } from "../../commands" +import type { IDb, IFs, ISystem, IToast, IUiIframe, IUiWorker } from "../client" + +// export function constructJarvisExtDBToServerDbAPI(db: JarvisExtDB): IDb { +// return { +// add: (data) => db.add(data), +// delete: (dataId) => db.delete(dataId), +// search: (searchParams) => db.search(searchParams), +// retrieveAll: (options) => db.retrieveAll(options), +// retrieveAllByType: (dataType) => db.retrieveAllByType(dataType), +// deleteAll: () => db.deleteAll(), +// update: (data) => db.update(data) +// } +// } diff --git a/packages/api/src/ui/server/deno.ts b/packages/api/src/ui/server/deno.ts new file mode 100644 index 0000000..8977668 --- /dev/null +++ b/packages/api/src/ui/server/deno.ts @@ -0,0 +1,432 @@ +import { join } from "@tauri-apps/api/path" +import { exists } from "@tauri-apps/plugin-fs" +import { difference } from "lodash" +import type { InternalSpawnOptions, SpawnOptions } from "tauri-plugin-shellx-api" +import { safeParse } from "valibot" +import { + PermissionScopeSchema, + ShellPermissionScopedSchema, + type ShellPermission, + type ShellPermissionScoped +} from "../../permissions/schema" +import { + AllPathAliases, + matchPathAndScope, + pathStartsWithAlias, + translateScopeToPath +} from "../../utils/path" +import { type DenoRunConfig } from "../client" + +/** + * + * @param userPermissionScopes + * @param program should be deno for now, may be I will support custom deno path in the future + * @param scriptPath absolute path to deno script + * @param config Deno config from extension API call + */ +export async function verifyDenoCmdPermission( + userPermissionScopes: ShellPermissionScoped[], + program: string, + scriptPath: string, + config: Partial & SpawnOptions, + extensionDir: string +) { + scriptPath = await translateScopeToPath(scriptPath, extensionDir) + if (!(await exists(scriptPath)) && config.cwd) { + scriptPath = await join(config.cwd, scriptPath) + } + const userDenoPerms = userPermissionScopes.filter( + (p) => p.permission && p.permission.startsWith("shell:deno:") + ) + const pathMatchedPerms: ShellPermissionScoped[] = [] + for (const perm of userDenoPerms) { + const safeParsed = safeParse(ShellPermissionScopedSchema, perm) + if (!safeParsed.success) { + throw new Error(`Invalid permission: ${perm}`) + } + const parsedPerm = safeParsed.output + // ignore the ones whose path doesn't match + let pathMatched = false + if (parsedPerm.allow) { + for (const allow of parsedPerm.allow) { + if (allow.path && (await matchPathAndScope(scriptPath, allow.path, extensionDir))) { + pathMatched = true + } + } + } + if (parsedPerm.deny) { + for (const deny of parsedPerm.deny) { + if (deny.path && (await matchPathAndScope(scriptPath, deny.path, extensionDir))) { + pathMatched = true + } + } + } + if (pathMatched) { + pathMatchedPerms.push(parsedPerm) + } + } + + /* -------------------- Start Verifying Deno Permissions -------------------- */ + const allowEnv: string[] = [] + let allowAllEnv = false + const denyEnv: string[] = [] + let denyAllEnv = false + const allowNet: string[] = [] + let allowAllNet = false + const denyNet: string[] = [] + let denyAllNet = false + let allowRead: string[] = [] + let allowAllRead = false + let denyRead: string[] = [] + let denyAllRead = false + let allowWrite: string[] = [] + let allowAllWrite = false + let denyWrite: string[] = [] + let denyAllWrite = false + const allowRun: string[] = [] + let allowAllRun = false + const denyRun: string[] = [] + let denyAllRun = false + const allowFfi: string[] = [] + let allowAllFfi = false + const denyFfi: string[] = [] + let denyAllFfi = false + const allowSys: string[] = [] + let allowAllSys = false + const denySys: string[] = [] + let denyAllSys = false + for (const perm of pathMatchedPerms) { + if (perm.allow) { + for (const allow of perm.allow) { + if (allow.env) { + if (allow.env === "*") { + allowAllEnv = true + } else { + allowEnv.push(...allow.env) + } + } + if (allow.net) { + if (allow.net === "*") { + allowAllNet = true + } else { + allowNet.push(...allow.net) + } + } + if (allow.read) { + if (allow.read === "*") { + allowAllRead = true + } else { + allowRead.push(...allow.read) + } + } + if (allow.write) { + if (allow.write === "*") { + allowAllWrite = true + } else { + allowWrite.push(...allow.write) + } + } + if (allow.run) { + if (allow.run === "*") { + allowAllRun = true + } else { + allowRun.push(...allow.run) + } + } + if (allow.ffi) { + if (allow.ffi === "*") { + allowAllFfi = true + } else { + allowFfi.push(...allow.ffi) + } + } + if (allow.sys) { + if (allow.sys === "*") { + allowAllSys = true + } else { + allowSys.push(...allow.sys) + } + } + } + } + if (perm.deny) { + for (const deny of perm.deny) { + if (deny.env) { + if (deny.env === "*") { + denyAllEnv = true + } else { + denyEnv.push(...deny.env) + } + } + if (deny.net) { + if (deny.net === "*") { + denyAllNet = true + } else { + denyNet.push(...deny.net) + } + } + if (deny.read) { + if (deny.read === "*") { + denyAllRead = true + } else { + denyRead.push(...deny.read) + } + } + if (deny.write) { + if (deny.write === "*") { + denyAllWrite = true + } else { + denyWrite.push(...deny.write) + } + } + if (deny.run) { + if (deny.run === "*") { + denyAllRun = true + } else { + denyRun.push(...deny.run) + } + } + if (deny.ffi) { + if (deny.ffi === "*") { + denyAllFfi = true + } else { + denyFfi.push(...deny.ffi) + } + } + if (deny.sys) { + if (deny.sys === "*") { + denyAllSys = true + } else { + denySys.push(...deny.sys) + } + } + } + } + } + /* ------------- Translate Paths for read and write Permissions ------------- */ + allowRead = await Promise.all( + allowRead.map(async (p) => + pathStartsWithAlias(p) ? await translateScopeToPath(p, extensionDir) : p + ) + ) + + if (config.allowRead) { + config.allowRead = await Promise.all( + config.allowRead.map(async (p) => await translateScopeToPath(p, extensionDir)) + ) + } + + if (config.denyRead) { + config.denyRead = await Promise.all( + config.denyRead.map(async (p) => await translateScopeToPath(p, extensionDir)) + ) + } + + if (config.allowWrite) { + config.allowWrite = await Promise.all( + config.allowWrite.map(async (p) => await translateScopeToPath(p, extensionDir)) + ) + } + + if (config.denyWrite) { + config.denyWrite = await Promise.all( + config.denyWrite.map(async (p) => await translateScopeToPath(p, extensionDir)) + ) + } + + denyRead = await Promise.all( + denyRead.map(async (p) => + pathStartsWithAlias(p) ? await translateScopeToPath(p, extensionDir) : p + ) + ) + + allowWrite = await Promise.all( + allowWrite.map(async (p) => + pathStartsWithAlias(p) ? await translateScopeToPath(p, extensionDir) : p + ) + ) + + denyWrite = await Promise.all( + denyWrite.map(async (p) => + pathStartsWithAlias(p) ? await translateScopeToPath(p, extensionDir) : p + ) + ) + + // now we have command requested permissions, we need to compare with permissions defined in manifest + /* ----------------------- Check Allow All Permissions ---------------------- */ + + if (config.allowAllEnv && !allowAllEnv) { + throw new Error("allowAllEnv is not allowed") + } + if (config.allowAllNet && !allowAllNet) { + throw new Error("allowAllNet is not allowed") + } + if (config.allowAllRead && !allowAllRead) { + throw new Error("allowAllRead is not allowed") + } + if (config.allowAllWrite && !allowAllWrite) { + throw new Error("allowAllWrite is not allowed") + } + if (config.allowAllRun && !allowAllRun) { + throw new Error("allowAllRun is not allowed") + } + if (config.allowAllFfi && !allowAllFfi) { + throw new Error("allowAllFfi is not allowed") + } + if (config.allowAllSys && !allowAllSys) { + throw new Error("allowAllSys is not allowed") + } + + if (difference(config.allowEnv, allowEnv).length > 0) { + throw new Error(`allowEnv is not allowed: ${difference(config.allowEnv, allowEnv)}`) + } + if (difference(config.allowNet, allowNet).length > 0) { + throw new Error(`allowNet is not allowed: ${difference(config.allowNet, allowNet)}`) + } + if (difference(config.allowRead, allowRead).length > 0) { + throw new Error(`allowRead is not allowed: ${difference(config.allowRead, allowRead)}`) + } + if (difference(config.allowWrite, allowWrite).length > 0) { + throw new Error(`allowWrite is not allowed: ${difference(config.allowWrite, allowWrite)}`) + } + if (difference(config.allowRun, allowRun).length > 0) { + throw new Error(`allowRun is not allowed: ${difference(config.allowRun, allowRun)}`) + } + if (difference(config.allowFfi, allowFfi).length > 0) { + throw new Error(`allowFfi is not allowed: ${difference(config.allowFfi, allowFfi)}`) + } + if (difference(config.allowSys, allowSys).length > 0) { + throw new Error(`allowSys is not allowed: ${difference(config.allowSys, allowSys)}`) + } +} + +/* -------------------------------------------------------------------------- */ +/* Translate Deno Command to a regular command */ +/* -------------------------------------------------------------------------- */ + +/** + * Translate deno script command to a regular shell command + * @param scriptPath + * @param config + * @returns + */ +export async function translateDenoCommand( + scriptPath: string, + config: Partial & SpawnOptions, + args: string[], + extensionDir: string +): Promise<{ + program: string + args: string[] + options: InternalSpawnOptions +}> { + const program = "deno" // TODO: support custom deno path + const shellArgs: string[] = ["run"] + /* ----------------------------------- Env ---------------------------------- */ + if (config.allowAllEnv) { + shellArgs.push("--allow-env") + } else if (config.allowEnv) { + const allowEnvStr = config.allowEnv.join(",") + shellArgs.push(`--allow-env=${allowEnvStr}`) + } + if (config.denyAllEnv) { + shellArgs.push("--deny-env") + } else if (config.denyEnv) { + const denyEnvStr = config.denyEnv.join(",") + shellArgs.push(`--deny-env=${denyEnvStr}`) + } + /* ----------------------------------- Net ---------------------------------- */ + if (config.allowAllNet) { + shellArgs.push("--allow-net") + } else if (config.allowNet) { + const allowNetStr = config.allowNet.join(",") + shellArgs.push(`--allow-net=${allowNetStr}`) + } + if (config.denyAllNet) { + shellArgs.push("--deny-net") + } else if (config.denyNet) { + const denyNetStr = config.denyNet.join(",") + shellArgs.push(`--deny-net=${denyNetStr}`) + } + /* ----------------------------------- Read ---------------------------------- */ + if (config.allowAllRead) { + shellArgs.push("--allow-read") + } else if (config.allowRead) { + const allowReadStr = config.allowRead.join(",") + shellArgs.push(`--allow-read=${allowReadStr}`) + } + if (config.denyAllRead) { + shellArgs.push("--deny-read") + } else if (config.denyRead) { + const denyReadStr = config.denyRead.join(",") + shellArgs.push(`--deny-read=${denyReadStr}`) + } + /* ----------------------------------- Write ---------------------------------- */ + if (config.allowAllWrite) { + shellArgs.push("--allow-write") + } else if (config.allowWrite) { + const allowWriteStr = config.allowWrite.join(",") + shellArgs.push(`--allow-write=${allowWriteStr}`) + } + if (config.denyAllWrite) { + shellArgs.push("--deny-write") + } else if (config.denyWrite) { + const denyWriteStr = config.denyWrite.join(",") + shellArgs.push(`--deny-write=${denyWriteStr}`) + } + /* ----------------------------------- Run ---------------------------------- */ + if (config.allowAllRun) { + shellArgs.push("--allow-run") + } else if (config.allowRun) { + const allowRunStr = config.allowRun.join(",") + shellArgs.push(`--allow-run=${allowRunStr}`) + } + if (config.denyAllRun) { + shellArgs.push("--deny-run") + } else if (config.denyRun) { + const denyRunStr = config.denyRun.join(",") + shellArgs.push(`--deny-run=${denyRunStr}`) + } + /* ----------------------------------- Ffi ---------------------------------- */ + if (config.allowAllFfi) { + shellArgs.push("--allow-ffi") + } else if (config.allowFfi) { + const allowFfiStr = config.allowFfi.join(",") + shellArgs.push(`--allow-ffi=${allowFfiStr}`) + } + if (config.denyAllFfi) { + shellArgs.push("--deny-ffi") + } else if (config.denyFfi) { + const denyFfiStr = config.denyFfi.join(",") + shellArgs.push(`--deny-ffi=${denyFfiStr}`) + } + /* ----------------------------------- Sys ---------------------------------- */ + if (config.allowAllSys) { + shellArgs.push("--allow-sys") + } else if (config.allowSys) { + const allowSysStr = config.allowSys.join(",") + shellArgs.push(`--allow-sys=${allowSysStr}`) + } + if (config.denyAllSys) { + shellArgs.push("--deny-sys") + } else if (config.denySys) { + const denySysStr = config.denySys.join(",") + shellArgs.push(`--deny-sys=${denySysStr}`) + } + /* ----------------------------------- Script ---------------------------------- */ + scriptPath = await translateScopeToPath(scriptPath, extensionDir) + shellArgs.push(scriptPath) + if (args) { + shellArgs.push(...args) + } + + return { + program, + args: shellArgs, + options: { + cwd: config.cwd, + env: config.env, + encoding: config.encoding + } + } +} diff --git a/packages/api/src/ui/server/event.ts b/packages/api/src/ui/server/event.ts new file mode 100644 index 0000000..b8fbd2b --- /dev/null +++ b/packages/api/src/ui/server/event.ts @@ -0,0 +1,58 @@ +/** + * `tauri-api-adapter` contains a `constructEventApi()`. + * We don't use that one because it exposes raw `listen()` and `emit()`, which may expose too much power to the client. + * Instead in this project, we define a custom `IEventServer` interface and `constructEventApi()` function that + * only exposes a limited set of events. + */ +import { listen, TauriEvent } from "@tauri-apps/api/event" +import { type EventPermission } from "../../permissions" +import { EventPermissionMap } from "../../permissions/permission-map" +import { checkPermission } from "../../utils/permission-check" +import type { DragDropPayload, DragEnterPayload, DragOverPayload, IEvent } from "../client" + +export function constructEventApi(permissions: EventPermission[]): IEvent { + return { + onDragDrop: (callback) => { + checkPermission(permissions, EventPermissionMap.onDragDrop) + listen(TauriEvent.DRAG_DROP, (e) => { + callback(e.payload) + }) + }, + onDragEnter: (callback) => { + checkPermission(permissions, EventPermissionMap.onDragEnter) + listen(TauriEvent.DRAG_ENTER, (e) => { + callback(e.payload) + }) + }, + onDragLeave: (callback) => { + checkPermission(permissions, EventPermissionMap.onDragLeave) + listen(TauriEvent.DRAG_LEAVE, (e) => { + callback() + }) + }, + onDragOver: (callback) => { + checkPermission(permissions, EventPermissionMap.onDragOver) + listen(TauriEvent.DRAG_OVER, (e) => { + callback(e.payload) + }) + }, + onWindowBlur: (callback) => { + checkPermission(permissions, EventPermissionMap.onWindowBlur) + listen(TauriEvent.WINDOW_BLUR, (e) => { + callback() + }) + }, + onWindowCloseRequested: (callback) => { + checkPermission(permissions, EventPermissionMap.onWindowCloseRequested) + listen(TauriEvent.WINDOW_CLOSE_REQUESTED, (e) => { + callback() + }) + }, + onWindowFocus: (callback) => { + checkPermission(permissions, EventPermissionMap.onWindowFocus) + listen(TauriEvent.WINDOW_FOCUS, (e) => { + callback() + }) + } + } +} diff --git a/packages/api/src/ui/server/fs.ts b/packages/api/src/ui/server/fs.ts new file mode 100644 index 0000000..6d9875d --- /dev/null +++ b/packages/api/src/ui/server/fs.ts @@ -0,0 +1,171 @@ +import { BaseDirectory } from "@tauri-apps/api/path" +import { + copyFile as fsCopyFile, + create as fsCreate, + exists as fsExists, + lstat as fsLstat, + mkdir as fsMkdir, + readDir as fsReadDir, + readFile as fsReadFile, + readTextFile as fsReadTextFile, + remove as fsRemove, + rename as fsRename, + stat as fsStat, + truncate as fsTruncate, + writeFile as fsWriteFile, + writeTextFile as fsWriteTextFile, + type CopyFileOptions, + type CreateOptions, + type ExistsOptions, + type MkdirOptions, + type ReadDirOptions, + type ReadFileOptions, + type RemoveOptions, + type RenameOptions, + type StatOptions, + type TruncateOptions, + type WriteFileOptions +} from "@tauri-apps/plugin-fs" +import { fileSearch, FileSearchParams } from "../../commands/fileSearch" +import { FsPermissionMap } from "../../permissions/permission-map" +import { + AllKunkunPermission, + type FsPermissionScoped, + type KunkunFsPermission, + type SystemPermission +} from "../../permissions/schema" +import { + combinePathAndBaseDir, + matchPathAndScope, + verifyGeneralPathScopedPermission +} from "../../utils/path" +import type { IFs } from "../client" + +/** + * `tauri-api-adapter` provides fs API + * In Kunkun I provide a more granular permission system and an extra file search API, so I rewrite the fs server API constructor + * @param permissions + * @returns + */ +export function constructFsApi(permissions: FsPermissionScoped[], extensionDir: string): IFs { + /** + * This is a helper function to simplify the creation of methods that takes one path and one options object + * @param requiredPermissions + * @param fn + * @returns + */ + const createMethod = Promise>( + requiredPermissions: KunkunFsPermission[], + fn: T + ) => { + return (path: string | URL, options?: { baseDir?: BaseDirectory }): ReturnType => + verifyGeneralPathScopedPermission( + requiredPermissions, + permissions, + path, + extensionDir, + options + ).then((result) => fn(path, options)) as ReturnType + } + return { + readDir: createMethod(FsPermissionMap.readDir, fsReadDir), + readFile: createMethod(FsPermissionMap.readFile, fsReadFile), + readTextFile: createMethod(FsPermissionMap.readTextFile, fsReadTextFile), + stat: createMethod(FsPermissionMap.stat, fsStat), + lstat: createMethod(FsPermissionMap.lstat, fsLstat), + exists: createMethod(FsPermissionMap.exists, fsExists), + mkdir: createMethod(FsPermissionMap.mkdir, fsMkdir), + create: createMethod(FsPermissionMap.create, fsCreate), + copyFile: (fromPath: string | URL, toPath: string | URL, options?: CopyFileOptions) => { + return fsStat(fromPath) + .then((oldPathStat) => { + const oldPathPermissions: KunkunFsPermission[] = ["fs:read"] + if (oldPathStat.isDirectory) { + oldPathPermissions.push("fs:read-dir") + } + return Promise.all([ + verifyGeneralPathScopedPermission( + oldPathPermissions, + permissions, + fromPath, + extensionDir, + { + baseDir: options?.fromPathBaseDir + } + ), + verifyGeneralPathScopedPermission(["fs:write"], permissions, toPath, extensionDir, { + baseDir: options?.toPathBaseDir + }) + ]) + }) + .then(() => fsCopyFile(fromPath, toPath, options)) + }, + remove: createMethod(FsPermissionMap.remove, fsRemove), + rename: async (oldPath: string | URL, newPath: string | URL, options?: RenameOptions) => { + return fsStat(oldPath) + .then((oldPathStat) => { + const oldPathPermissions: KunkunFsPermission[] = ["fs:read"] + if (oldPathStat.isDirectory) { + oldPathPermissions.push("fs:read-dir") + } + return Promise.all([ + verifyGeneralPathScopedPermission( + oldPathPermissions, + permissions, + oldPath, + extensionDir, + { + baseDir: options?.oldPathBaseDir + } + ), + verifyGeneralPathScopedPermission(["fs:write"], permissions, newPath, extensionDir, { + baseDir: options?.oldPathBaseDir + }) + ]) + }) + .then(() => fsRename(oldPath, newPath, options)) + }, + truncate: (path: string | URL, len?: number, options?: TruncateOptions) => + verifyGeneralPathScopedPermission( + FsPermissionMap.truncate, + permissions, + path, + extensionDir, + options + ).then(() => fsTruncate(path, len, options)), + writeFile: (path: string | URL, data: Uint8Array, options?: WriteFileOptions) => + verifyGeneralPathScopedPermission( + FsPermissionMap.truncate, + permissions, + path, + extensionDir, + options + ).then(() => fsWriteFile(path, data, options)), + writeTextFile: (path: string | URL, data: string, options?: WriteFileOptions) => + verifyGeneralPathScopedPermission( + FsPermissionMap.truncate, + permissions, + path, + extensionDir, + options + ).then(() => fsWriteTextFile(path, data, options)), + fileSearch: ( + searchParams: Omit & { + hidden?: boolean + ignore_case?: boolean + } + ) => { + return Promise.all( + // TODO: first verify all search locations are allowed, for now, recursive search is allowed even if scope allows one level only + searchParams.locations.map((loc) => + verifyGeneralPathScopedPermission( + FsPermissionMap.fileSearch, + permissions, + loc, + extensionDir + ) + ) + ).then(() => fileSearch(searchParams)) + } + } +} diff --git a/packages/api/src/ui/server/index.ts b/packages/api/src/ui/server/index.ts new file mode 100644 index 0000000..bba3c62 --- /dev/null +++ b/packages/api/src/ui/server/index.ts @@ -0,0 +1,206 @@ +import { + constructClipboardApi, + constructDialogApi, + constructFetchApi, + // constructFsApi, // a local constructFsApi is defined + constructLoggerApi, + constructNetworkApi, + constructNotificationApi, + constructOsApi, + // constructPathApi, + // constructShellApi, // a local custom constructShellApi is defined + constructSystemInfoApi, + constructUpdownloadApi, + type IClipboard, + type IDialog, + type IFetchInternal, + type ILogger, + type INetwork, + type INotification, + type IOs, + type IPath, + type IShell, + type IShellServer, + type ISystemInfo, + type IUpdownload +} from "tauri-api-adapter" +import { type IFetch } from "tauri-api-adapter/client" +import { + checkPermission, + type ClipboardPermission, + type DialogPermission, + type FetchPermission, + type NetworkPermission, + type NotificationPermission, + type OsPermission, + // type ShellPermission, + type SystemInfoPermission, + type UpdownloadPermission +} from "tauri-api-adapter/permissions" +import { + AllKunkunPermission, + type EventPermission, + type FsPermissionScoped, + type KunkunFsPermission, + type OpenPermissionScoped, + type SecurityPermission, + type ShellPermission, + type ShellPermissionScoped, + type SystemPermission +} from "../../permissions" +import type { IEvent, IFs, IOpen, ISecurity, ISystem, IToast, IUtils } from "../client" +// import type { IDbServer } from "./db" +import { constructEventApi } from "./event" +import { constructFsApi } from "./fs" +import { constructOpenApi } from "./open" +import { constructPathApi } from "./path" +import { constructSecurityAPI } from "./security" +import type { IUiIframeServer1 } from "./server-types" +// import type { IFsServer, ISystemServer } from "./server-types" +import { constructShellApi } from "./shell" +import { constructSystemApi } from "./system" +import { + constructToastApi + // type IToastServer +} from "./toast" +import { + constructIframeUiApi + // type IUiIframeServer, + // type IUiWorkerServer +} from "./ui" +import { constructUtilsApi } from "./utils" + +// export type { IDbServer } from "./db" +export { constructFsApi } from "./fs" +export { constructSystemApi } from "./system" +export { + constructToastApi + // type IToastServer +} from "./toast" +export { + constructIframeUiApi + // type IUiIframeServer, + // type IUiWorkerServer +} from "./ui" + +// export type IJarvisFullAPI = +// // IFullAPI & +// ISystemServer & IToastServer & IDbServer & IUiWorkerServer & IUiIframeServer & IFsServer // IFsServer will override some methods in IFullAPI, it's fine because it's a superset +type AllScopedPermissions = FsPermissionScoped | OpenPermissionScoped | ShellPermissionScoped +type AllPermissions = AllKunkunPermission | AllScopedPermissions +function getStringPermissions(permissions: AllPermissions[]): AllKunkunPermission[] { + return permissions.filter((p) => typeof p === "string") as AllKunkunPermission[] +} + +function getObjectPermissions(permissions: AllPermissions[]): AllScopedPermissions[] { + return permissions.filter((p) => typeof p !== "string") +} + +export type IKunkunFullServerAPI = { + clipboard: IClipboard + fetch: IFetchInternal + dialog: IDialog + event: IEvent + fs: IFs + log: ILogger + path: Omit // BaseDirectory is only on Client side, not on Server side + notification: INotification + os: IOs + updownload: IUpdownload + sysInfo: ISystemInfo + network: INetwork + open: IOpen + system: ISystem + toast: IToast + shell: IShellServer + iframeUi: IUiIframeServer1 + utils: IUtils + security: ISecurity +} + +/** + * Construct Jarvis Server API with permissions constraints + * @param permissions all permissions user has + * @param extPath absolute path to the extension + * @returns + */ +export function constructJarvisServerAPIWithPermissions( + permissions: AllPermissions[], + extPath: string +): IKunkunFullServerAPI { + return { + clipboard: constructClipboardApi( + getStringPermissions(permissions).filter((p) => + p.startsWith("clipboard:") + ) as ClipboardPermission[] + ), + fetch: constructFetchApi( + getStringPermissions(permissions).filter((p) => p.startsWith("fetch:")) as FetchPermission[] + ), + dialog: constructDialogApi( + getStringPermissions(permissions).filter((p) => p.startsWith("dialog:")) as DialogPermission[] + ), + event: constructEventApi( + getStringPermissions(permissions).filter((p) => p.startsWith("event:")) as EventPermission[] + ), // this one is not from tauri-api-adapter, it's a custom one defined in this project, only limited events are exposed + fs: constructFsApi( + (getObjectPermissions(permissions) as FsPermissionScoped[]).filter((p) => + p.permission.startsWith("fs:") + ), + extPath + ), + log: constructLoggerApi(), + path: constructPathApi(extPath), + notification: constructNotificationApi( + getStringPermissions(permissions).filter((p) => + p.startsWith("notification:") + ) as NotificationPermission[] + ), + os: constructOsApi( + getStringPermissions(permissions).filter((p) => p.startsWith("os:")) as OsPermission[] + ), + updownload: constructUpdownloadApi( + getStringPermissions(permissions).filter((p) => + p.startsWith("updownload:") + ) as UpdownloadPermission[] + ), + network: constructNetworkApi( + getStringPermissions(permissions).filter((p) => + p.startsWith("network:") + ) as NetworkPermission[] + ), + open: constructOpenApi( + (getObjectPermissions(permissions) as OpenPermissionScoped[]).filter((p) => + p.permission.startsWith("open:") + ), + extPath + ), + sysInfo: constructSystemInfoApi( + getStringPermissions(permissions).filter((p) => + p.startsWith("system-info:") + ) as SystemInfoPermission[] + ), + system: constructSystemApi( + getStringPermissions(permissions).filter((p) => p.startsWith("system:")) as SystemPermission[] + ), + toast: constructToastApi(), + shell: constructShellApi( + [ + ...(getStringPermissions(permissions).filter((p) => + p.startsWith("shell:") + ) as ShellPermission[]), + ...(getObjectPermissions(permissions) as ShellPermissionScoped[]).filter((p) => + p.permission.startsWith("shell:") + ) + ], + extPath + ), + iframeUi: constructIframeUiApi(), + utils: constructUtilsApi(), + security: constructSecurityAPI( + getStringPermissions(permissions).filter((p) => + p.startsWith("security:") + ) as SecurityPermission[] + ) + } +} diff --git a/packages/api/src/ui/server/open.ts b/packages/api/src/ui/server/open.ts new file mode 100644 index 0000000..dfb8e1d --- /dev/null +++ b/packages/api/src/ui/server/open.ts @@ -0,0 +1,100 @@ +import { exists, stat } from "@tauri-apps/plugin-fs" +import { minimatch } from "minimatch" +import { open } from "tauri-plugin-shellx-api" +import { flatten, parse, pipe, safeParse, string, url, type InferOutput } from "valibot" +import type { OpenPermissionScoped } from "../../permissions" +import { + combinePathAndBaseDir, + matchPathAndScope, + pathStartsWithAlias, + translateScopeToPath, + verifyScopedPermission +} from "../../utils/path" +import type { IOpen } from "../client" + +const UrlSchema = pipe(string("A URL must be string."), url("The URL is badly formatted.")) + +export function constructOpenApi(permissions: OpenPermissionScoped[], extDir: string): IOpen { + return { + url: async (url: string) => { + const parseResult = safeParse(UrlSchema, url) + if (parseResult.success) { + } else { + console.error(flatten(parseResult.issues)) + throw new Error(`URL is Invalid: ${url}`) + } + // permission check + if ( + await verifyScopedPermission( + permissions.filter((p) => p.permission === "open:url"), + parseResult.output, + "url", + extDir + ) + ) { + open(parseResult.output) + } else { + throw new Error( + `Permission denied to open URL: ${parseResult.output}, add "open:url" permission with scope rule to match URL pattern.` + ) + } + }, + file: async (path: string) => { + // check if path is a file and exists + let p = parse(string(), path) + if (pathStartsWithAlias(p)) { + p = await translateScopeToPath(p, extDir) + } + console.log("open file path", path) + if (!(await exists(p))) { + throw new Error(`File does not exist: ${path}`) + } else { + const s = await stat(p) + if (!s.isFile) { + throw new Error(`Path is not a file: ${path}`) + } + } + // permission check + if ( + await verifyScopedPermission( + permissions.filter((p) => p.permission === "open:file"), + p, + "path", + extDir + ) + ) { + open(p) + } else { + throw new Error( + `Permission denied to open file: ${path}, add "open:file" permission with scope to match path pattern.` + ) + } + }, + folder: async (path: string) => { + // check if path is a directory and exists + const p = parse(string(), path) + if (!(await exists(p))) { + throw new Error(`Directory does not exist: ${path}`) + } else { + const s = await stat(p) + if (!s.isDirectory) { + throw new Error(`Path is not a directory: ${path}`) + } + } + if ( + await verifyScopedPermission( + permissions.filter((p) => p.permission === "open:folder"), + p, + "path", + extDir + ) + ) { + open(p) + } else { + throw new Error( + `Permission denied to open field: ${path}, add "open:folder" permission with scope to match path pattern.` + ) + } + } + } +} diff --git a/packages/api/src/ui/server/path.ts b/packages/api/src/ui/server/path.ts new file mode 100644 index 0000000..b0645e2 --- /dev/null +++ b/packages/api/src/ui/server/path.ts @@ -0,0 +1,37 @@ +import * as path from "@tauri-apps/api/path" +import { exists, mkdir } from "@tauri-apps/plugin-fs" +import { constructPathApi as constructTauriPathApi } from "tauri-api-adapter" +import type { IPath } from "../client" + +export async function constructExtensionSupportDir(extPath: string) { + const appDataDir = await path.appDataDir() + const extSupportDir = await path.join( + appDataDir, + "extensions_support", + await path.basename(extPath) + ) + if (extPath.startsWith(appDataDir)) { + return extSupportDir + } else { + const extSupportDir = await path.join(extPath, "extensions_support") + if (await exists(extSupportDir)) { + return extSupportDir + } else { + await mkdir(extSupportDir, { recursive: true }) + return extSupportDir + } + } +} + +/** + * Return type omits BaseDirectory because it's only used on client side + * @param extPath absolute path to the extension + * @returns + */ +export function constructPathApi(extPath: string): Omit { + return { + ...constructTauriPathApi(), + extensionDir: () => Promise.resolve(extPath), + extensionSupportDir: () => constructExtensionSupportDir(extPath) + } +} diff --git a/packages/api/src/ui/server/security.ts b/packages/api/src/ui/server/security.ts new file mode 100644 index 0000000..f436dc2 --- /dev/null +++ b/packages/api/src/ui/server/security.ts @@ -0,0 +1,53 @@ +import { macSecurity } from "@kksh/api/commands" +import { Command, open } from "tauri-plugin-shellx-api" +import * as v from "valibot" +import { SecurityPermissionMap, type SecurityPermission } from "../../permissions" +import { checkPermission } from "../../utils/permission-check" +import { MacSecurityOptions, type ISecurity } from "../client" + +export function constructSecurityAPI(permissions: SecurityPermission[]): ISecurity { + return { + mac: { + revealSecurityPane: (privacyOption?: MacSecurityOptions) => { + checkPermission( + permissions, + SecurityPermissionMap.mac.revealSecurityPane + ) + if (privacyOption) { + const parsedOption = v.parse(MacSecurityOptions, privacyOption) + return open( + `x-apple.systempreferences:com.apple.preference.security?Privacy_${parsedOption}` + ) + } else { + return open("x-apple.systempreferences:com.apple.preference.security") + } + }, + verifyFingerprint: async () => { + return macSecurity.verifyAuth() + }, + requestScreenCapturePermission: async () => { + checkPermission( + permissions, + SecurityPermissionMap.mac.requestScreenCapturePermission + ) + return macSecurity.requestScreenCaptureAccess() + }, + checkScreenCapturePermission: async () => { + checkPermission( + permissions, + SecurityPermissionMap.mac.checkScreenCapturePermission + ) + return macSecurity.checkScreenCaptureAccess() + }, + resetPermission: (privacyOption: MacSecurityOptions) => { + const parsedOption = v.parse(MacSecurityOptions, privacyOption) + const cmd = Command.create("tccutil", ["reset", parsedOption, "sh.kunkun.desktop"]) + return cmd.execute().then((res) => { + if (res.code !== 0) { + throw new Error(`Failed to reset permission: ${res.stderr}`) + } + }) + } + } + } +} diff --git a/packages/api/src/ui/server/server-types.ts b/packages/api/src/ui/server/server-types.ts new file mode 100644 index 0000000..7092231 --- /dev/null +++ b/packages/api/src/ui/server/server-types.ts @@ -0,0 +1,38 @@ +// import type { IEvent, IFs, ISystem } from "../client" + +import { type IShellServer as IShellServer1 } from "tauri-api-adapter" +import type { + ChildProcess, + CommandEvent, + InternalSpawnOptions, + IOPayload, + SpawnOptions +} from "tauri-plugin-shellx-api" +import type { DenoRunConfig, IUiIframe } from "../client" + +export type IShellServer = IShellServer1 & { + denoExecute( + scriptPath: string, + config: Partial & SpawnOptions, + args: string[] + ): Promise> + denoRawSpawn( + scriptPath: string, + config: Partial & SpawnOptions, + args: string[], + cb: (evt: CommandEvent) => void + ): Promise + recordSpawnedProcess(pid: number): Promise +} + +// This will be implemented in the @kksh/api package +export type IUiIframeServer1 = Pick< + IUiIframe, + "startDragging" | "toggleMaximize" | "internalToggleMaximize" +> +// This interface will be implemented in iframe-ext.vue where iframe is loaded and API is exposed +// because these API dependes on the context of the page +export type IUiIframeServer2 = Omit< + IUiIframe, + "registerDragRegion" | "internalToggleMaximize" | "toggleMaximize" | "startDragging" +> diff --git a/packages/api/src/ui/server/shell.ts b/packages/api/src/ui/server/shell.ts new file mode 100644 index 0000000..71b761e --- /dev/null +++ b/packages/api/src/ui/server/shell.ts @@ -0,0 +1,292 @@ +import { emitKillProcessEvent } from "@kksh/api/events" +import { Channel, invoke } from "@tauri-apps/api/core" +import { emitTo } from "@tauri-apps/api/event" +import { getCurrentWindow } from "@tauri-apps/api/window" +import { + type ChildProcess, + type CommandEvent, + type InternalSpawnOptions, + type IOPayload +} from "tauri-plugin-shellx-api" +import { RECORD_EXTENSION_PROCESS_EVENT, type IRecordExtensionProcessEvent } from "../../events" +import { ShellPermissionMap } from "../../permissions/permission-map" +import { type ShellPermission, type ShellPermissionScoped } from "../../permissions/schema" +import { verifyScopedPermission } from "../../utils/path" +import type { DenoRunConfig } from "../client" +import { translateDenoCommand, verifyDenoCmdPermission } from "./deno" +import type { IShellServer } from "./server-types" + +function matchRegexArgs(args: string[], regexes: string[]): boolean { + if (args.length !== regexes.length) { + return false + } + for (let i = 0; i < args.length; i++) { + let regex = regexes[i]! + if (!regex.startsWith("^")) { + regex = `^${regex}` + } + if (!regex.endsWith("$")) { + regex = `${regex}$` + } + if (!new RegExp(regex).test(args[i]!)) { + return false + } + } + return true +} + +async function verifyShellCmdPermission( + requiredPermissions: ShellPermission[], + userPermissionScopes: ShellPermissionScoped[], + program: string, + args: string[] +): Promise { + for (const permission of userPermissionScopes) { + if (requiredPermissions.includes(permission.permission)) { + for (const deny of permission.deny || []) { + if (deny.cmd && deny.cmd.program === program && matchRegexArgs(args, deny.cmd.args || [])) { + return Promise.reject("Shell Command Permission Denied by deny rule") + } + } + for (const allow of permission.allow || []) { + if ( + allow.cmd && + allow.cmd.program === program && + matchRegexArgs(args, allow.cmd.args || []) + ) { + return Promise.resolve() + } + } + } + } + return Promise.reject("Shell Command Permission Denied, no allow rule found") +} + +/** + * `tauri-api-adapter` provides shell API. + * In kunkun I provide a more granular permission system and extra shell script execution APIs, so I rewrite the shell server API constructor + * @param permissions + * @param extPath absolute path to the extension + * @returns + */ +export function constructShellApi( + permissions: (ShellPermissionScoped | ShellPermission)[], + extPath: string +): IShellServer { + const stringPermissiongs = permissions.filter((p) => typeof p === "string") as ShellPermission[] + const objectPermissions = permissions.filter( + (p) => typeof p !== "string" + ) as ShellPermissionScoped[] + + async function execute( + program: string, + args: string[], + options: InternalSpawnOptions + ): Promise> { + await verifyShellCmdPermission(ShellPermissionMap.execute, objectPermissions, program, args) + return invoke>("plugin:shellx|execute", { + program: program, + args: args, + options: options + }) + } + function kill(pid: number) { + if (!stringPermissiongs.some((p) => ShellPermissionMap.kill.includes(p))) + return Promise.reject( + new Error(`Permission denied. Requires one of ${ShellPermissionMap.kill}`) + ) + return invoke("plugin:shellx|kill", { + cmd: "killChild", + pid: pid + }).then(() => { + emitKillProcessEvent(pid) + }) + } + function stdinWrite(buffer: string | number[], pid: number) { + if (!stringPermissiongs.some((p) => ShellPermissionMap.stdinWrite.includes(p))) + return Promise.reject( + new Error(`Permission denied. Requires one of ${ShellPermissionMap.stdinWrite}`) + ) + return invoke("plugin:shellx|stdin_write", { + buffer: buffer, + pid: pid + }) + } + async function open(path: string, openWith?: string) { + // TODO: consider adding a base dir option like the fs api's + // this should throw if permission is denied + if ( + await verifyScopedPermission( + objectPermissions.filter( + (p) => p.permission === "shell:open" || p.permission === "shell:all" + ), + path, + "url", + extPath + ) + ) { + return open(path, openWith) + } else { + throw new Error(`Permission denied to open file: ${path}`) + } + } + // shellOpen: verifyShellCmdPermission(["shell:open"], permissions)(shellOpen), + async function rawSpawn( + program: string, + args: string[], + options: InternalSpawnOptions, + cb: (evt: CommandEvent) => void + ) { + await verifyShellCmdPermission(ShellPermissionMap.rawSpawn, objectPermissions, program, args) + const onEvent = new Channel>() + onEvent.onmessage = cb + return invoke("plugin:shellx|spawn", { + program: program, + args: args, + options: options, + onEvent + }) + } + async function executeBashScript(script: string): Promise> { + await verifyShellCmdPermission(ShellPermissionMap.execute, objectPermissions, "bash", [ + "-c", + script + ]) + return executeBashScript(script) + } + async function executePowershellScript(script: string): Promise> { + await verifyShellCmdPermission( + ShellPermissionMap.executePowershellScript, + objectPermissions, + "powershell", + ["-Command", script] + ) + return executePowershellScript(script) + } + async function executeAppleScript(script: string): Promise> { + await verifyShellCmdPermission( + ShellPermissionMap.executeAppleScript, + objectPermissions, + "osascript", + ["-e", script] + ) + return executeAppleScript(script) + } + async function executePythonScript(script: string): Promise> { + await verifyShellCmdPermission( + ShellPermissionMap.executePythonScript, + objectPermissions, + "python", + ["-c", script] + ) + return executePythonScript(script) + } + async function executeZshScript(script: string): Promise> { + await verifyShellCmdPermission(ShellPermissionMap.executeZshScript, objectPermissions, "zsh", [ + "-c", + script + ]) + return executeZshScript(script) + } + async function executeNodeScript(script: string): Promise> { + await verifyShellCmdPermission( + ShellPermissionMap.executeNodeScript, + objectPermissions, + "node", + ["-e", script] + ) + return executeNodeScript(script) + } + async function hasCommand(command: string): Promise { + // check if command is clean, check if it's a single command without arguments or semicolons with regex. + if (!/^[a-zA-Z0-9_-]+$/.test(command)) { + return Promise.reject(new Error("Invalid command")) + } + return hasCommand(command) + } + async function likelyOnWindows(): Promise { + return likelyOnWindows() + } + + return { + async recordSpawnedProcess(pid: number): Promise { + // get window label + const curWin = await getCurrentWindow() + console.log("recordSpawnedProcess", pid, curWin.label) + await emitTo("main", RECORD_EXTENSION_PROCESS_EVENT, { + windowLabel: curWin.label, + pid + } satisfies IRecordExtensionProcessEvent) + // TODO: record process in a store + return Promise.resolve() + }, + async denoExecute( + scriptPath: string, + config: DenoRunConfig, + args1: string[] + ): Promise> { + await verifyDenoCmdPermission( + objectPermissions.filter((p) => p.permission.startsWith("shell:deno:")), + "deno", + scriptPath, + config, + extPath + ) + const { program, args, options } = await translateDenoCommand( + scriptPath, + config, + args1, + extPath + ) + console.log("denoExecute", program, args, options) + return invoke>("plugin:shellx|execute", { + program, + args, + options + }) + // return execute(program, args, options) + }, + async denoRawSpawn( + scriptPath: string, + config: DenoRunConfig, + args1: string[], + cb: (evt: CommandEvent) => void + ): Promise { + await verifyDenoCmdPermission( + objectPermissions.filter((p) => p.permission.startsWith("shell:deno:")), + "deno", + scriptPath, + config, + extPath + ) + const { program, args, options } = await translateDenoCommand( + scriptPath, + config, + args1, + extPath + ) + const onEvent = new Channel>() + onEvent.onmessage = cb + return invoke("plugin:shellx|spawn", { + program, + args, + options, + onEvent + }) + // return rawSpawn(program, args, options, cb) + }, + execute, + kill, + stdinWrite, + open, + rawSpawn, + executeBashScript, + executePowershellScript, + executeAppleScript, + executePythonScript, + executeZshScript, + executeNodeScript, + hasCommand, + likelyOnWindows + } +} diff --git a/packages/api/src/ui/server/system.ts b/packages/api/src/ui/server/system.ts new file mode 100644 index 0000000..a86027f --- /dev/null +++ b/packages/api/src/ui/server/system.ts @@ -0,0 +1,150 @@ +import { checkPermission } from "tauri-api-adapter/permissions" +import { + ejectAllDisks, + emptyTrash, + getFrontmostApp, + getSelectedFilesInFileExplorer, + hideAllAppsExceptFrontmost, + logoutUser, + mute, + openTrash, + quitAllApps, + reboot, + setVolume, + setVolumeTo0, + setVolumeTo25, + setVolumeTo50, + setVolumeTo75, + setVolumeTo100, + showDesktop, + shutdown, + sleep, + sleepDisplays, + toggleBluetooth, + toggleHiddenFiles, + toggleMute, + toggleStageManager, + toggleSystemAppearance, + turnVolumeDown, + turnVolumeUp, + unmute +} from "../../commands/system" +import { + AllKunkunPermission, + type FsPermissionScoped, + type KunkunFsPermission, + type SystemPermission +} from "../../permissions" +import { SystemPermissionMap } from "../../permissions/permission-map" +import type { ISystem } from "../client" + +export function constructSystemApi(permissions: SystemPermission[]): ISystem { + return { + openTrash: checkPermission( + SystemPermissionMap.openTrash, + permissions + )(openTrash), + emptyTrash: checkPermission( + SystemPermissionMap.emptyTrash, + permissions + )(emptyTrash), + shutdown: checkPermission( + SystemPermissionMap.shutdown, + permissions + )(shutdown), + reboot: checkPermission(SystemPermissionMap.reboot, permissions)(reboot), + sleep: checkPermission(SystemPermissionMap.sleep, permissions)(sleep), + toggleSystemAppearance: checkPermission( + SystemPermissionMap.toggleSystemAppearance, + permissions + )(toggleSystemAppearance), + showDesktop: checkPermission( + SystemPermissionMap.showDesktop, + permissions + )(showDesktop), + quitAllApps: checkPermission( + SystemPermissionMap.quitAllApps, + permissions + )(quitAllApps), + sleepDisplays: checkPermission( + SystemPermissionMap.sleepDisplays, + permissions + )(sleepDisplays), + setVolume: checkPermission( + SystemPermissionMap.setVolume, + permissions + )(setVolume), + setVolumeTo0: checkPermission( + SystemPermissionMap.setVolumeTo0, + permissions + )(setVolumeTo0), + setVolumeTo25: checkPermission( + SystemPermissionMap.setVolumeTo25, + permissions + )(setVolumeTo25), + setVolumeTo50: checkPermission( + SystemPermissionMap.setVolumeTo50, + permissions + )(setVolumeTo50), + setVolumeTo75: checkPermission( + SystemPermissionMap.setVolumeTo75, + permissions + )(setVolumeTo75), + setVolumeTo100: checkPermission( + SystemPermissionMap.setVolumeTo100, + permissions + )(setVolumeTo100), + turnVolumeUp: checkPermission( + SystemPermissionMap.turnVolumeUp, + permissions + )(turnVolumeUp), + turnVolumeDown: checkPermission( + SystemPermissionMap.turnVolumeDown, + permissions + )(turnVolumeDown), + toggleStageManager: checkPermission( + SystemPermissionMap.toggleStageManager, + permissions + )(toggleStageManager), + toggleBluetooth: checkPermission([], permissions)(toggleBluetooth), + toggleHiddenFiles: checkPermission( + SystemPermissionMap.toggleHiddenFiles, + permissions + )(toggleHiddenFiles), + ejectAllDisks: checkPermission( + SystemPermissionMap.ejectAllDisks, + permissions + )(ejectAllDisks), + logoutUser: checkPermission( + SystemPermissionMap.logoutUser, + permissions + )(logoutUser), + toggleMute: checkPermission( + SystemPermissionMap.toggleMute, + permissions + )(toggleMute), + mute: checkPermission(SystemPermissionMap.mute, permissions)(mute), + unmute: checkPermission(SystemPermissionMap.unmute, permissions)(unmute), + getFrontmostApp: checkPermission( + SystemPermissionMap.getFrontmostApp, + permissions + )(getFrontmostApp), + hideAllAppsExceptFrontmost: checkPermission( + SystemPermissionMap.hideAllAppsExceptFrontmost, + permissions + )(hideAllAppsExceptFrontmost), + getSelectedFilesInFileExplorer: checkPermission( + SystemPermissionMap.getSelectedFilesInFileExplorer, + permissions + )(getSelectedFilesInFileExplorer) + } +} + +// all enabled by default +// export const defaultSystemApi = constructSystemApi([ +// "system:volumn", +// "system:boot", +// "system:disk", +// "system:apps", +// "system:fs" +// ]) diff --git a/packages/api/src/ui/server/toast.ts b/packages/api/src/ui/server/toast.ts new file mode 100644 index 0000000..bcb33df --- /dev/null +++ b/packages/api/src/ui/server/toast.ts @@ -0,0 +1,56 @@ +import { toast } from "svelte-sonner" +import type { + GeneralToast, + GeneralToastParams, + IDb, + IFs, + ISystem, + IToast, + IUiIframe, + IUiWorker +} from "../client" + +async function constructToast( + fn: + | typeof toast.message + | typeof toast.info + | typeof toast.success + | typeof toast.warning + | typeof toast.error, + message: string, + options?: GeneralToastParams, + action?: () => void +): Promise { + await fn(message, { + ...options, + action: + action && options?.actionLabel + ? { + label: options?.actionLabel ?? "Action", + onClick: () => { + action() + } + } + : undefined + }) +} + +export function constructToastApi(): IToast { + return { + message: async (message: string, options?: GeneralToastParams, action?: () => void) => { + constructToast(toast.message, message, options, action) + }, + info: async (message: string, options?: GeneralToastParams, action?: () => void) => { + constructToast(toast.info, message, options, action) + }, + success: async (message: string, options?: GeneralToastParams, action?: () => void) => { + constructToast(toast.success, message, options, action) + }, + warning: async (message: string, options?: GeneralToastParams, action?: () => void) => { + constructToast(toast.warning, message, options, action) + }, + error: async (message: string, options?: GeneralToastParams, action?: () => void) => { + constructToast(toast.error, message, options, action) + } + } +} diff --git a/packages/api/src/ui/server/ui.ts b/packages/api/src/ui/server/ui.ts new file mode 100644 index 0000000..cb8890d --- /dev/null +++ b/packages/api/src/ui/server/ui.ts @@ -0,0 +1,22 @@ +import { invoke } from "@tauri-apps/api/core" +import { getCurrentWindow } from "@tauri-apps/api/window" +import type { IUiIframeServer1 } from "./server-types" + +/** + * Other APIs will be constructed in main window as they are used to manipulate UI directly + * We can't access UI from here + * @returns + */ +export function constructIframeUiApi(): IUiIframeServer1 { + return { + startDragging: () => { + return getCurrentWindow().startDragging() + }, + toggleMaximize: () => { + return getCurrentWindow().toggleMaximize() + }, + internalToggleMaximize: () => { + return invoke("plugin:window|internal_toggle_maximize") + } + } +} diff --git a/packages/api/src/ui/server/utils.ts b/packages/api/src/ui/server/utils.ts new file mode 100644 index 0000000..a45d0f7 --- /dev/null +++ b/packages/api/src/ui/server/utils.ts @@ -0,0 +1,20 @@ +import { plistToJson } from "../../commands/utils" +import type { IUtils } from "../client" + +export function constructUtilsApi(): IUtils { + return { + plist: { + // build: (data: any) => { + // return Promise.resolve(plist.build(data)) + // }, + /** + * parse plist content to json string + * @param plistContent + * @returns + */ + parse: (plistContent: string) => { + return plistToJson(plistContent) + } + } + } +} diff --git a/packages/api/src/ui/window.ts b/packages/api/src/ui/window.ts new file mode 100644 index 0000000..4b8b22d --- /dev/null +++ b/packages/api/src/ui/window.ts @@ -0,0 +1,68 @@ +import * as _webviewApis from "@tauri-apps/api/webview" +import * as _webviewWindowApis from "@tauri-apps/api/webviewWindow" +import * as _windowApis from "@tauri-apps/api/window" + +export { getCurrentWindow, getAllWindows, currentMonitor } from "@tauri-apps/api/window" +export { WebviewWindow } from "@tauri-apps/api/webviewWindow" + +type Unlisten = () => void + +export function closeWindow() { + return _windowApis.getCurrentWindow().close() +} + +/** + * Force current window to close + * @returns Promise + */ +export function destroyWindow() { + return _windowApis.getCurrentWindow().destroy() +} + +export function windowLabelExists(label: string) { + return _windowApis.getAllWindows().then((wins) => wins.some((w) => w.label === label)) +} + +export function getWindowByLabel(label: string) { + return _windowApis.getAllWindows().then((wins) => wins.find((w) => w.label === label)) +} + +export function closeMainWindow() { + return _windowApis.getAllWindows().then((wins) => wins.find((w) => w.label === "main")?.close()) +} + +export function hideWindow(windowLabel: string) { + return _windowApis + .getAllWindows() + .then((wins) => wins.find((w) => w.label === windowLabel)?.hide()) +} + +export function hideMainWindow() { + return hideWindow("main") +} + +export function showWindow(windowLabel: string) { + return _windowApis + .getAllWindows() + .then((wins) => wins.find((w) => w.label === windowLabel)?.show()) +} + +export function showMainWindow() { + return showWindow("main") +} + +/** + * Run this function to close the current window when the specified key is pressed + * An unlistener is returned to stop listening, but not necessary as the window will be destroyed + * @param key The key to listen for, default is Escape + * @returns + */ +export function destroyWindowOnKeyPressed(key: string = "Escape"): Unlisten { + function quitOn(e: KeyboardEvent) { + if (e.key === key) { + destroyWindow() + } + } + document.addEventListener("keydown", quitOn) + return () => document.removeEventListener("keydown", quitOn) +} diff --git a/packages/api/src/ui/worker/components/action.ts b/packages/api/src/ui/worker/components/action.ts new file mode 100644 index 0000000..75255ac --- /dev/null +++ b/packages/api/src/ui/worker/components/action.ts @@ -0,0 +1,99 @@ +import { NodeName, NodeNameEnum } from "../../../models/constants" +import * as ActionSchema from "../schema/action" +import type { IconConstructorPatch, OmitNodeName, ReplaceIcon } from "./common" +import { Icon } from "./icon" +import type { IComponent } from "./interfaces" + +export class Action implements ActionSchema.Action, IComponent { + nodeName: NodeName = NodeNameEnum.Action + icon?: Icon + title: string + value: string + + constructor(model: IconConstructorPatch) { + this.icon = model.icon + this.title = model.title + this.value = model.value + } + + toModel(): { + nodeName: NodeNameEnum + title: string + value: string + icon?: Icon + } { + return { + nodeName: this.nodeName, + title: this.title, + value: this.value, + icon: this.icon + } + } +} + +// export class ActionPanelSection +// implements ActionSchema.ActionPanelSection, IComponent +// { +// nodeName: NodeName = NodeNameEnum.ActionPanelSection +// title: string +// actions: Action[] + +// constructor(model: OmitNodeName & { actions: Action[] }) { +// this.title = model.title +// this.actions = model.actions +// } + +// toModel(): ActionSchema.ActionPanelSection { +// return { +// nodeName: this.nodeName, +// title: this.title, +// actions: this.actions.map((action) => action.toModel()) +// } +// } +// } + +// export class ActionPanelSubmenu +// implements ActionSchema.ActionPaneSubmenu, IComponent +// { +// nodeName: NodeName = NodeNameEnum.ActionPanelSubmenu +// title: string +// actions: Action[] + +// constructor(model: OmitNodeName & { actions: Action[] }) { +// this.title = model.title +// this.actions = model.actions +// } + +// toModel(): ActionSchema.ActionPaneSubmenu { +// return { +// nodeName: this.nodeName, +// title: this.title, +// actions: this.actions.map((action) => action.toModel()) +// } +// } +// } + +export class ActionPanel implements ActionSchema.ActionPanel, IComponent { + nodeName: NodeName = NodeNameEnum.ActionPanel + title?: string + items: Array + + constructor( + model: OmitNodeName & { + items: Array + } + ) { + this.title = model.title + this.items = model.items + } + + toModel(): ActionSchema.ActionPanel { + return { + nodeName: this.nodeName, + title: this.title, + items: this.items.map((item) => item.toModel()) + } + } +} diff --git a/packages/api/src/ui/worker/components/common.ts b/packages/api/src/ui/worker/components/common.ts new file mode 100644 index 0000000..1f81db3 --- /dev/null +++ b/packages/api/src/ui/worker/components/common.ts @@ -0,0 +1,5 @@ +import { Icon } from "./icon" + +export type OmitNodeName = Omit +export type ReplaceIcon = T & { icon?: Icon } +export type IconConstructorPatch = ReplaceIcon> diff --git a/packages/api/src/ui/worker/components/form-view.ts b/packages/api/src/ui/worker/components/form-view.ts new file mode 100644 index 0000000..5f25a7c --- /dev/null +++ b/packages/api/src/ui/worker/components/form-view.ts @@ -0,0 +1,160 @@ +import { FormNodeName, FormNodeNameEnum, NodeName } from "../../../models" +import * as FormSchema from "../schema/form" +import type { OmitNodeName } from "./common" +import { type IComponent } from "./interfaces" + +export abstract class BaseField implements FormSchema.BaseField, IComponent { + nodeName: FormNodeName = FormNodeNameEnum.Base + key: string + label?: string + hideLabel?: boolean + placeholder?: string + optional?: boolean + description?: string + default?: any + + constructor(model: OmitNodeName) { + this.key = model.key + this.key = model.key + this.label = model.label + this.hideLabel = model.hideLabel + this.placeholder = model.placeholder + this.optional = model.optional + this.description = model.description + this.default = model.default + } + + toModel(): FormSchema.BaseField { + return { + nodeName: this.nodeName, + key: this.key, + label: this.label, + hideLabel: this.hideLabel, + placeholder: this.placeholder, + optional: this.optional, + description: this.description, + default: this.default + } + } +} + +export class InputField + extends BaseField + implements FormSchema.InputField, IComponent +{ + nodeName: FormNodeName = FormNodeNameEnum.Input + component?: "textarea" | "default" + + constructor(model: OmitNodeName) { + super(model) + this.component = model.component + } + + toModel(): FormSchema.InputField { + return { + ...super.toModel(), + component: this.component + } + } +} + +export class NumberField + extends BaseField + implements FormSchema.NumberField, IComponent +{ + nodeName: FormNodeNameEnum.Number = FormNodeNameEnum.Number +} + +export class SelectField + extends BaseField + implements FormSchema.SelectField, IComponent +{ + nodeName: FormNodeName = FormNodeNameEnum.Select + options: string[] + + constructor(model: OmitNodeName) { + super(model) + this.options = model.options + } + + toModel(): FormSchema.SelectField { + return { + ...super.toModel(), + options: this.options + } + } +} + +export class BooleanField + extends BaseField + implements FormSchema.BooleanField, IComponent +{ + nodeName: FormNodeName = FormNodeNameEnum.Boolean + component?: "checkbox" | "switch" + + constructor(model: OmitNodeName) { + super(model) + this.component = model.component ?? "checkbox" + } + + toModel(): FormSchema.BooleanField { + return { + ...super.toModel(), + component: this.component + } + } +} + +export class DateField + extends BaseField + implements FormSchema.DateField, IComponent +{ + nodeName: FormNodeName = FormNodeNameEnum.Date +} + +export type AllFormFields = InputField | NumberField | SelectField | BooleanField | DateField + +export class ArrayField + extends BaseField + implements FormSchema.ArrayField, IComponent +{ + nodeName: FormNodeName = FormNodeNameEnum.Array + content: AllFormFields + + constructor(model: OmitNodeName) { + super(model) + this.content = model.content + } + + toModel(): FormSchema.ArrayField { + return { + ...super.toModel(), + content: this.content.toModel() + } + } +} + +export class Form implements IComponent { + nodeName: FormNodeName = FormNodeNameEnum.Form + fields: (AllFormFields | Form)[] + key: string + title?: string + description?: string + submitBtnText?: string + + constructor(model: OmitNodeName) { + this.fields = model.fields + this.key = model.key + } + + toModel(): FormSchema.Form { + return { + nodeName: this.nodeName, + key: this.key, + title: this.title, + description: this.description, + submitBtnText: this.submitBtnText, + fields: this.fields.map((field) => field.toModel()) + } + } +} diff --git a/packages/api/src/ui/worker/components/icon.ts b/packages/api/src/ui/worker/components/icon.ts new file mode 100644 index 0000000..205a3ae --- /dev/null +++ b/packages/api/src/ui/worker/components/icon.ts @@ -0,0 +1,22 @@ +import { NodeName, NodeNameEnum } from "../../../models/constants" +import type { IconNode, IconType, Icon as TIcon } from "../../../models/icon" +import { type IComponent } from "./interfaces" + +export class Icon implements IconNode, IComponent { + nodeName: NodeName = NodeNameEnum.Icon + type: IconType + value: string + + constructor(model: TIcon) { + this.type = model.type + this.value = model.value + } + + toModel(): IconNode { + return { + nodeName: this.nodeName, + type: this.type, + value: this.value + } + } +} diff --git a/packages/api/src/ui/worker/components/index.ts b/packages/api/src/ui/worker/components/index.ts new file mode 100644 index 0000000..8d9cda8 --- /dev/null +++ b/packages/api/src/ui/worker/components/index.ts @@ -0,0 +1,6 @@ +export { type IComponent } from "./interfaces" +export * from "./icon" +export * as List from "./list-view" +export * as Action from "./action" +export * as Form from "./form-view" +export { Markdown } from "./markdown" diff --git a/packages/api/src/ui/worker/components/interfaces.ts b/packages/api/src/ui/worker/components/interfaces.ts new file mode 100644 index 0000000..f280068 --- /dev/null +++ b/packages/api/src/ui/worker/components/interfaces.ts @@ -0,0 +1,6 @@ +import type { FormNodeName, NodeName } from "../../../models/constants" + +export interface IComponent { + nodeName: NodeName | FormNodeName + toModel(): T +} diff --git a/packages/api/src/ui/worker/components/list-view.ts b/packages/api/src/ui/worker/components/list-view.ts new file mode 100644 index 0000000..449f247 --- /dev/null +++ b/packages/api/src/ui/worker/components/list-view.ts @@ -0,0 +1,404 @@ +import { NodeName, NodeNameEnum } from "../../../models/constants" +import type { Color } from "../../../models/styles" +import * as ListSchema from "../schema/list" +import * as Action from "./action" +import type { IconConstructorPatch, OmitNodeName, ReplaceIcon } from "./common" +import { Icon } from "./icon" +import { type IComponent } from "./interfaces" +import { Markdown } from "./markdown" + +export class EmptyView implements ListSchema.EmptyView, IComponent { + nodeName: NodeName = NodeNameEnum.EmptyView + title?: string + description?: string + icon?: Icon + + constructor(model: IconConstructorPatch) { + this.title = model.title + this.description = model.description + this.icon = model.icon + } + + toModel(): ListSchema.EmptyView { + return { + nodeName: this.nodeName, + title: this.title, + description: this.description, + icon: this.icon?.toModel() + } + } +} + +export class DropdownItem implements ListSchema.DropdownItem, IComponent { + nodeName: NodeName = NodeNameEnum.DropdownItem + title: string + value: string + icon?: Icon + keywords?: string[] + + constructor(model: IconConstructorPatch) { + this.title = model.title + this.value = model.value + this.icon = model.icon + this.keywords = model.keywords + } + + toModel(): ListSchema.DropdownItem { + return { + nodeName: this.nodeName, + title: this.title, + value: this.value, + icon: this.icon?.toModel(), + keywords: this.keywords + } + } +} + +export class DropdownSection + implements ListSchema.DropdownSection, IComponent +{ + nodeName: NodeName = NodeNameEnum.DropdownSection + title: string + items: DropdownItem[] + + constructor(model: OmitNodeName & { items: DropdownItem[] }) { + this.title = model.title + this.items = model.items + } + + toModel(): ListSchema.DropdownSection { + return { + nodeName: this.nodeName, + title: this.title, + items: this.items.map((item) => item.toModel()) + } + } +} + +export class Dropdown implements ListSchema.Dropdown, IComponent { + nodeName: NodeName = NodeNameEnum.Dropdown + tooltip: string + sections: DropdownSection[] + defaultValue: string + + constructor(model: IconConstructorPatch & { sections: DropdownSection[] }) { + this.tooltip = model.tooltip + this.sections = model.sections + this.defaultValue = model.defaultValue + } + + toModel(): ListSchema.Dropdown { + return { + nodeName: this.nodeName, + tooltip: this.tooltip, + sections: this.sections.map((section) => section.toModel()), + defaultValue: this.defaultValue + } + } +} + +export class ItemAccessory + implements ListSchema.ItemAccessory, IComponent +{ + nodeName: NodeName = NodeNameEnum.ListItemAccessory + tag?: string | { color: Color; text: string } + text?: string | { color: Color; text: string } + date?: Date | { color: Color; text: Date } + icon?: Icon + tooltip?: string + + constructor(model: IconConstructorPatch) { + this.tag = model.tag + this.text = model.text + this.date = model.date + this.icon = model.icon + this.tooltip = model.tooltip + } + + toModel(): ListSchema.ItemAccessory { + return { + nodeName: this.nodeName, + tag: this.tag, + text: this.text, + date: this.date, + icon: this.icon?.toModel(), + tooltip: this.tooltip + } + } +} + +export class ItemDetailMetadataLabel + implements ListSchema.ItemDetailMetadataLabel, IComponent +{ + nodeName: NodeNameEnum.ListItemDetailMetadataLabel = NodeNameEnum.ListItemDetailMetadataLabel + title: string + icon?: Icon + text?: string | { color: Color; text: string } + + constructor(model: IconConstructorPatch) { + this.title = model.title + this.icon = model.icon + this.text = model.text + } + + toModel(): ListSchema.ItemDetailMetadataLabel { + return { + nodeName: this.nodeName, + title: this.title, + icon: this.icon?.toModel(), + text: this.text + } + } +} + +export class ItemDetailMetadataLink + implements ListSchema.ItemDetailMetadataLink, IComponent +{ + nodeName: NodeNameEnum.ListItemDetailMetadataLink = NodeNameEnum.ListItemDetailMetadataLink + title: string + text: string + url: string + + constructor(model: OmitNodeName) { + this.title = model.title + this.text = model.text + this.url = model.url + } + + toModel(): ListSchema.ItemDetailMetadataLink { + return { + nodeName: this.nodeName, + title: this.title, + text: this.text, + url: this.url + } + } +} + +export class ItemDetailMetadataTagListItem + implements + ListSchema.ItemDetailMetadataTagListItem, + IComponent +{ + nodeName: NodeNameEnum.ListItemDetailMetadataTagListItem = + NodeNameEnum.ListItemDetailMetadataTagListItem + text?: string + color?: Color + icon?: Icon + + constructor(model: IconConstructorPatch) { + this.text = model.text + this.color = model.color + this.icon = model.icon + } + + toModel(): ListSchema.ItemDetailMetadataTagListItem { + return { + nodeName: this.nodeName, + text: this.text, + color: this.color + } + } +} + +export class ItemDetailMetadataTagList + implements ListSchema.ItemDetailMetadataTagList, IComponent +{ + nodeName: NodeNameEnum.ListItemDetailMetadataTagList = NodeNameEnum.ListItemDetailMetadataTagList + title: string + tags: ItemDetailMetadataTagListItem[] + + constructor( + model: OmitNodeName & { + tags: ItemDetailMetadataTagListItem[] + } + ) { + this.title = model.title + this.tags = model.tags + } + + toModel(): ListSchema.ItemDetailMetadataTagList { + return { + nodeName: this.nodeName, + title: this.title, + tags: this.tags.map((tag) => tag.toModel()) + } + } +} + +export class ItemDetailMetadataSeparator + implements + ListSchema.ItemDetailMetadataSeparator, + IComponent +{ + nodeName: NodeNameEnum.ListItemDetailMetadataSeparator = + NodeNameEnum.ListItemDetailMetadataSeparator + + toModel(): ListSchema.ItemDetailMetadataSeparator { + return { + nodeName: this.nodeName + } + } +} + +export class ItemDetailMetadata + implements ListSchema.ItemDetailMetadata, IComponent +{ + nodeName: NodeNameEnum.ListItemDetailMetadata = NodeNameEnum.ListItemDetailMetadata + items: ( + | ItemDetailMetadataLabel + | ItemDetailMetadataLink + | ItemDetailMetadataTagList + | ItemDetailMetadataSeparator + )[] + + constructor( + items: ( + | ItemDetailMetadataLabel + | ItemDetailMetadataLink + | ItemDetailMetadataTagList + | ItemDetailMetadataSeparator + )[] + ) { + this.items = items + } + + toModel(): ListSchema.ItemDetailMetadata { + return { + nodeName: this.nodeName, + items: this.items.map((item) => item.toModel()) + } + } +} + +export class ItemDetail implements ListSchema.ItemDetail, IComponent { + nodeName: NodeNameEnum.ListItemDetail = NodeNameEnum.ListItemDetail + children: (ItemDetailMetadata | Markdown)[] + width?: number + + constructor( + model: OmitNodeName + ) { + this.children = model.children + this.width = model.width + } + + toModel(): ListSchema.ItemDetail { + return { + nodeName: this.nodeName, + children: this.children.map((child) => child.toModel()), + width: this.width + } + } +} + +export class Item implements ListSchema.Item, IComponent { + nodeName: NodeNameEnum.ListItem = NodeNameEnum.ListItem + title: string + value: string + subTitle?: string + accessories?: ItemAccessory[] + icon?: Icon + keywords?: string[] + defaultAction?: string + actions?: Action.ActionPanel + + constructor( + model: OmitNodeName & { + accessories?: ItemAccessory[] + icon?: Icon + actions?: Action.ActionPanel + } + ) { + this.title = model.title + this.value = model.value + this.actions = model.actions + this.defaultAction = model.defaultAction + this.subTitle = model.subTitle + this.accessories = model.accessories + this.icon = model.icon + this.keywords = model.keywords + } + + toModel(): ListSchema.Item { + return { + nodeName: this.nodeName, + title: this.title, + value: this.value, + defaultAction: this.defaultAction, + actions: this.actions?.toModel(), + subTitle: this.subTitle, + accessories: this.accessories?.map((accessory) => accessory.toModel()), + icon: this.icon?.toModel(), + keywords: this.keywords + } + } +} + +export class Section implements ListSchema.Section, IComponent { + nodeName: NodeNameEnum.ListSection = NodeNameEnum.ListSection + title?: string + items: Item[] + + constructor(model: OmitNodeName & { items: Item[] }) { + this.title = model.title + this.items = model.items + } + + toModel(): ListSchema.Section { + return { + nodeName: this.nodeName, + title: this.title, + items: this.items.map((item) => item.toModel()) + } + } +} + +export class List implements ListSchema.List, IComponent { + nodeName: NodeNameEnum.List = NodeNameEnum.List + sections?: Section[] + items?: Item[] + detail?: ItemDetail + filter: "none" | "default" + inherits?: ListSchema.ListInheritOptions[] + actions?: Action.ActionPanel + defaultAction?: string + /** + * - `updateDetailOnly`: If true, the list view inherits previous list view items and only updates the detail view. + * This is useful when you want to update the detail view without changing the list view items. + * For example, when user goes over item by item, you can update the detail view without resending the items data. This could be more efficient when there are many items. + * @param model + */ + constructor( + model: Omit, "filter" | "updateDetailOnly"> & { + sections?: Section[] + items?: Item[] + detail?: ItemDetail + filter?: "none" | "default" + inherits?: ListSchema.ListInheritOptions[] + actions?: Action.ActionPanel + defaultAction?: string + } + ) { + this.sections = model.sections + this.items = model.items + this.detail = model.detail + this.filter = model.filter ?? "default" + this.inherits = model.inherits ?? [] + this.actions = model.actions + this.defaultAction = model.defaultAction + } + + toModel(): ListSchema.List { + return { + nodeName: this.nodeName, + sections: this.sections?.map((section) => section.toModel()), + items: this.items?.map((item) => item.toModel()), + filter: this.filter, + detail: this.detail?.toModel(), + inherits: this.inherits, + actions: this.actions?.toModel(), + defaultAction: this.defaultAction + } + } +} diff --git a/packages/api/src/ui/worker/components/markdown.ts b/packages/api/src/ui/worker/components/markdown.ts new file mode 100644 index 0000000..8e7a4ad --- /dev/null +++ b/packages/api/src/ui/worker/components/markdown.ts @@ -0,0 +1,19 @@ +import { NodeName, NodeNameEnum } from "../../../models/constants" +import { type Markdown as MarkdownSchema } from "../schema/markdown" +import { type IComponent } from "./interfaces" + +export class Markdown implements MarkdownSchema, IComponent { + nodeName: NodeName = NodeNameEnum.Markdown + content: string + + constructor(content: string) { + this.content = content + } + + toModel(): MarkdownSchema { + return { + nodeName: this.nodeName, + content: this.content + } + } +} diff --git a/packages/api/src/ui/worker/ext.ts b/packages/api/src/ui/worker/ext.ts new file mode 100644 index 0000000..3e5da56 --- /dev/null +++ b/packages/api/src/ui/worker/ext.ts @@ -0,0 +1,49 @@ +export abstract class WorkerExtension { + /* -------------------------------------------------------------------------- */ + /* Common */ + /* -------------------------------------------------------------------------- */ + searchTerm: string = "" + highlightedListItemValue?: string + /** + * Load the extension. Initialize the extension. + * Will be called once when the extension is first loaded. + */ + abstract load(): Promise + onSearchTermChange(term: string): Promise { + this.searchTerm = term + return Promise.resolve() + } + + onActionSelected(value: string): Promise { + return Promise.resolve() + } + onEnterPressedOnSearchBar(): Promise { + return Promise.resolve() + } + + onBeforeGoBack(): Promise { + return Promise.resolve() + } + /* -------------------------------------------------------------------------- */ + /* List */ + /* -------------------------------------------------------------------------- */ + onListItemSelected(value: string): Promise { + return Promise.resolve() + } + + onListScrolledToBottom(): Promise { + return Promise.resolve() + } + + onHighlightedListItemChanged(value: string): Promise { + this.highlightedListItemValue = value + return Promise.resolve() + } + + /* -------------------------------------------------------------------------- */ + /* Form */ + /* -------------------------------------------------------------------------- */ + onFormSubmit(value: Record): Promise { + return Promise.resolve() + } +} diff --git a/packages/api/src/ui/worker/index.ts b/packages/api/src/ui/worker/index.ts new file mode 100644 index 0000000..424b22a --- /dev/null +++ b/packages/api/src/ui/worker/index.ts @@ -0,0 +1,115 @@ +/* -------------------------------------------------------------------------- */ +/* API */ +/* -------------------------------------------------------------------------- */ +import { wrap, type Endpoint, type Remote } from "@huakunshen/comlink" +import type { + IClipboard, + IDialog, + // IEventInternal, + IFetchInternal, + // IFs, + ILogger, + INetwork, + INotification, + IOs, + // IPath, + IShellInternal, + ISystemInfo, + IUpdownload +} from "tauri-api-adapter" +import { + constructFetchAPI, + // constructPathAPI, + // constructShellAPI, + constructUpdownloadAPI +} from "tauri-api-adapter/client" +import { constructEventAPI } from "../api/event" +import { constructPathAPI } from "../api/path" +import { constructShellAPI } from "../api/shell" +import { constructToastAPI } from "../api/toast" +import type { + IApp, + IDb, + IEvent, + IFs, + IOpen, + IPath, + ISecurity, + ISystem, + IToast, + IUiWorker, + IUtils +} from "../client" +import type { IShellServer } from "../server/server-types" + +export { expose, wrap } from "@huakunshen/comlink" +export { WorkerExtension } from "./ext" +export type { IDb } from "../client" +/** + * For the APIs annotated with "inherit from tauri-api-adapter", they inherit the client API completely from tauri-api-adapter + * There may be server API changes for them, but the client API can be inherited + */ +type API = { + db: Remote // for kunkun + system: Remote // for kunkun + open: Remote // for kunkun + clipboard: Remote // inherit from tauri-api-adapter + dialog: Remote // inherit from tauri-api-adapter + fetch: Remote // inherit from tauri-api-adapter + event: Remote // for kunkun, override tauri-api-adapter's event API, expose only specified event, disallow, emit and listen + fs: Remote // customized for kunkun, add file search API on top of tauri-api-adapter's fs API + log: Remote // inherit from tauri-api-adapter + notification: Remote // inherit from tauri-api-adapter + toast: Remote // for kunkun + os: Remote // inherit from tauri-api-adapter + path: Remote // inherit from tauri-api-adapter + shell: Remote // inherit from tauri-api-adapter + updownload: IUpdownload // inherit from tauri-api-adapter + sysInfo: Remote // inherit from tauri-api-adapter + network: Remote // inherit from tauri-api-adapter + workerUi: Remote // for kunkun + security: ISecurity // for kunkun + utils: IUtils // for kunkun + app: IApp +} + +const _api = wrap(globalThis as Endpoint) as unknown as API +export const event = constructEventAPI(_api.event) // this is different from event api from tauri-api-adapter +export const fetch = constructFetchAPI(_api.fetch) +export const path = constructPathAPI(_api.path) +export const shell = constructShellAPI(_api.shell) +export const toast = constructToastAPI(_api.toast) +export const updownload = constructUpdownloadAPI(_api.updownload) +export const { + db, + os, + clipboard, + dialog, + fs, + log, + notification, + sysInfo, + network, + system, + open, + utils, + app, + security, + workerUi: ui +} = _api +export { Child, RPCChannel } from "../api/shell" + +/* -------------------------------------------------------------------------- */ +/* UI Component Schema */ +/* -------------------------------------------------------------------------- */ +export { type IComponent } from "./components" +export * as ListSchema from "./schema/list" +export * as FormSchema from "./schema/form" +export { Markdown as MarkdownSchema } from "./schema/markdown" + +export { List, Action, Form, Markdown } from "./components" +export * from "../../models/styles" +export { Icon } from "./components/icon" +export { IconEnum, IconType, IconNode } from "../../models/icon" +export * as schema from "./schema" +export { NodeName, NodeNameEnum, FormNodeName, FormNodeNameEnum } from "../../models/constants" diff --git a/packages/api/src/ui/worker/schema/__test__/form.test.ts b/packages/api/src/ui/worker/schema/__test__/form.test.ts new file mode 100644 index 0000000..f63227f --- /dev/null +++ b/packages/api/src/ui/worker/schema/__test__/form.test.ts @@ -0,0 +1,85 @@ +import { describe, expect, test } from "bun:test" +import * as v from "valibot" +import { FormNodeNameEnum } from "../../../../models/constants" +import * as FormSchema from "../form" + +const nestedForm: FormSchema.Form = { + nodeName: FormNodeNameEnum.Form, + key: "form1", + fields: [ + { + nodeName: FormNodeNameEnum.Form, + key: "form2", + fields: [ + { + nodeName: FormNodeNameEnum.Input, + key: "input1" + } + ] + }, + { + nodeName: FormNodeNameEnum.Number, + key: "num1" + }, + { + key: "array1", + nodeName: FormNodeNameEnum.Array, + content: { + key: "input2", + nodeName: FormNodeNameEnum.Number + } + } + ] +} + +const arr: FormSchema.ArrayField = { + nodeName: FormNodeNameEnum.Array, + key: "array1", + content: { + nodeName: FormNodeNameEnum.Input, + key: "input1" + } +} + +describe("Verify Nested Form Schema Parsing", () => { + test("Array Field", () => { + const result = v.parse(FormSchema.ArrayField, arr) + expect(result.content.key).toEqual("input1") + }) + test("Nested Form", () => { + const result = v.parse(FormSchema.Form, nestedForm) + expect(result.fields[0]?.key).toEqual("form2") + expect((result.fields[0] as FormSchema.Form).fields[0]?.key).toEqual("input1") + // console.log(result) + expect((result.fields[2] as FormSchema.ArrayField).content).toBeDefined() + }) + + test("ext", () => { + const InputS = v.object({ name: v.string(), value: v.string() }) + const NumberS = v.object({ + name: v.number(), + value: v.string(), + content: v.object({ + name: v.string() + }) + }) + const FormS = v.object({ + name: v.string(), + arr: v.array(v.union([InputS, NumberS])) + }) + const result = v.parse(FormS, { + name: "name", + arr: [ + { name: "name", value: "value" }, + { + name: 1, + value: "value", + content: { + name: "name" + } + } + ] + }) + // console.log(result) + }) +}) diff --git a/packages/api/src/ui/worker/schema/action.ts b/packages/api/src/ui/worker/schema/action.ts new file mode 100644 index 0000000..d62ba5c --- /dev/null +++ b/packages/api/src/ui/worker/schema/action.ts @@ -0,0 +1,53 @@ +import { + array, + boolean, + date, + enum_, + hexColor, + literal, + nullable, + object, + optional, + pipe, + string, + union, + type InferOutput +} from "valibot" +import { NodeName, NodeNameEnum } from "../../../models/constants" +import { Icon } from "../../../models/icon" +import { Color } from "../../../models/styles" + +export const Action = object({ + nodeName: NodeName, + icon: optional(Icon), + title: string(), + value: string() + // shortcut: optional(string()) +}) +export type Action = InferOutput + +// export const ActionPaneSubmenu = object({ +// nodeName: NodeName, +// title: string(), +// actions: array(Action) +// }) +// export type ActionPaneSubmenu = InferOutput + +// export const ActionPanelSection = object({ +// nodeName: NodeName, +// title: string(), +// actions: array(Action) +// }) +// export type ActionPanelSection = InferOutput + +export const ActionPanel = object({ + nodeName: NodeName, + title: optional(string()), + items: array( + union([ + // ActionPanelSection, ActionPaneSubmenu, + Action + ]) + ) +}) +export type ActionPanel = InferOutput diff --git a/packages/api/src/ui/worker/schema/form.ts b/packages/api/src/ui/worker/schema/form.ts new file mode 100644 index 0000000..711dc82 --- /dev/null +++ b/packages/api/src/ui/worker/schema/form.ts @@ -0,0 +1,144 @@ +import { + any, + array, + boolean, + lazy, + literal, + number, + object, + optional, + parse, + required, + string, + union, + type GenericSchema, + type InferOutput +} from "valibot" +import { FormNodeName, FormNodeNameEnum } from "../../../models/constants" + +// boolean (checkbox, switch) +// date (date picker) +// enum (select, radio group) +// number (input) +// string (input, textfield) +// file (file) + +// This is only for string inputs (types that outputs string) +export const InputTypes = union([ + literal("color"), + literal("date"), + literal("datetime-local"), + literal("month"), + literal("number"), + literal("password"), + literal("text"), + literal("url"), + literal("week"), + literal("time"), + literal("search") +]) +export type InputTypes = InferOutput +// export const InputProps = object({ +// placeholder: optional(string()), +// // defaultValue: optional(string()), +// // these types can be properly rendered and output a string +// type: optional(InputTypes), +// showLabel: optional(boolean()), +// required: optional(boolean()) +// }) + +export const BaseField = object({ + nodeName: FormNodeName, + key: string(), + label: optional(string()), + hideLabel: optional(boolean()), + placeholder: optional(string()), + optional: optional(boolean()), + description: optional(string()), + default: optional(any()) +}) +export type BaseField = InferOutput + +/* -------------------------------------------------------------------------- */ +/* Input Element */ +/* -------------------------------------------------------------------------- */ +export const InputField = object({ + ...BaseField.entries, + type: optional(InputTypes), + component: optional(union([literal("textarea"), literal("default")])) +}) +export type InputField = InferOutput + +/* -------------------------------------------------------------------------- */ +/* Number */ +/* -------------------------------------------------------------------------- */ +export const NumberField = object({ + ...BaseField.entries, + nodeName: FormNodeName +}) +export type NumberField = InferOutput + +/* -------------------------------------------------------------------------- */ +/* Select */ +/* -------------------------------------------------------------------------- */ +// with zod enum +export const SelectField = object({ + ...BaseField.entries, + options: array(string()) +}) +export type SelectField = InferOutput + +/* -------------------------------------------------------------------------- */ +/* Boolean */ +/* -------------------------------------------------------------------------- */ +export const BooleanField = object({ + ...BaseField.entries, + component: optional(union([literal("checkbox"), literal("switch")])) +}) +export type BooleanField = InferOutput + +/* -------------------------------------------------------------------------- */ +/* Date */ +/* -------------------------------------------------------------------------- */ +export const DateField = object({ + ...BaseField.entries +}) +export type DateField = InferOutput + +export const AllFormFields = union([InputField, NumberField, SelectField, BooleanField, DateField]) + +/* -------------------------------------------------------------------------- */ +/* Array */ +/* -------------------------------------------------------------------------- */ +export const ArrayField = object({ + ...BaseField.entries, + content: AllFormFields +}) +export type ArrayField = InferOutput + +/* -------------------------------------------------------------------------- */ +/* Form */ +/* -------------------------------------------------------------------------- */ +export const FormField = union([ + ArrayField, // this must be placed first, otherwise its content field won't be parsed + InputField, + NumberField, + SelectField, + BooleanField, + DateField +]) +export type FormField = InferOutput +// export type Form = InferOutput +export type Form = { + nodeName: FormNodeName + title?: string + description?: string + submitBtnText?: string + key: string + fields: (FormField | Form)[] +} +export const Form: GenericSchema
= object({ + nodeName: FormNodeName, + key: string(), + fields: array(union([lazy(() => Form), FormField])) +}) diff --git a/packages/api/src/ui/worker/schema/index.ts b/packages/api/src/ui/worker/schema/index.ts new file mode 100644 index 0000000..986c19b --- /dev/null +++ b/packages/api/src/ui/worker/schema/index.ts @@ -0,0 +1 @@ +export * from "../../../models/icon" diff --git a/packages/api/src/ui/worker/schema/keyboard.ts b/packages/api/src/ui/worker/schema/keyboard.ts new file mode 100644 index 0000000..ffd3884 --- /dev/null +++ b/packages/api/src/ui/worker/schema/keyboard.ts @@ -0,0 +1,15 @@ +import { + array, + boolean, + date, + enum_, + hexColor, + literal, + nullable, + object, + optional, + pipe, + string, + union, + type InferOutput +} from "valibot" diff --git a/packages/api/src/ui/worker/schema/list.ts b/packages/api/src/ui/worker/schema/list.ts new file mode 100644 index 0000000..2472e17 --- /dev/null +++ b/packages/api/src/ui/worker/schema/list.ts @@ -0,0 +1,203 @@ +import { + array, + boolean, + date, + enum_, + hexColor, + literal, + nullable, + number, + object, + optional, + pipe, + string, + union, + type InferOutput +} from "valibot" +import { NodeName, NodeNameEnum } from "../../../models/constants" +import { Icon } from "../../../models/icon" +import { Color } from "../../../models/styles" +import * as ActionSchema from "./action" +import { Markdown } from "./markdown" + +/* -------------------------------------------------------------------------- */ +/* Empty View */ +/* -------------------------------------------------------------------------- */ +export const EmptyView = object({ + nodeName: NodeName, + title: optional(string()), + description: optional(string()), + icon: optional(Icon) +}) +export type EmptyView = InferOutput + +/* -------------------------------------------------------------------------- */ +/* Dropdown */ +/* -------------------------------------------------------------------------- */ +export const DropdownItem = object({ + nodeName: NodeName, + title: string(), + value: string(), + icon: optional(Icon), + keywords: optional(array(string())) +}) +export const DropdownSection = object({ + nodeName: NodeName, + title: string(), + items: array(DropdownItem) +}) +export const Dropdown = object({ + nodeName: NodeName, + tooltip: string(), + sections: array(DropdownSection), + defaultValue: string() +}) +export type DropdownItem = InferOutput +export type DropdownSection = InferOutput +export type Dropdown = InferOutput + +/* -------------------------------------------------------------------------- */ +/* List */ +/* -------------------------------------------------------------------------- */ +export const ItemAccessory = object({ + nodeName: NodeName, + tag: optional( + union([ + string(), + object({ + color: Color, + text: string() + }) + ]) + ), + text: optional(union([string(), object({ color: Color, text: string() })])), + date: optional(union([date(), object({ color: Color, text: date() })])), + icon: optional(Icon), + tooltip: optional(string()) +}) +export type ItemAccessory = InferOutput + +export const ItemDetailMetadataLabel = object({ + nodeName: literal(NodeNameEnum.ListItemDetailMetadataLabel), + title: string(), + icon: optional(Icon), + text: optional( + union([ + string(), + object({ + color: Color, + text: string() + }) + ]) + ) +}) +export type ItemDetailMetadataLabel = InferOutput + +export const ItemDetailMetadataLink = object({ + nodeName: literal(NodeNameEnum.ListItemDetailMetadataLink), + title: string(), + text: string(), + url: string() +}) +export type ItemDetailMetadataLink = InferOutput + +export const ItemDetailMetadataTagListItem = object({ + nodeName: literal(NodeNameEnum.ListItemDetailMetadataTagListItem), + text: optional(string()), + color: optional(Color) +}) +export type ItemDetailMetadataTagListItem = InferOutput + +export const ItemDetailMetadataTagList = object({ + nodeName: literal(NodeNameEnum.ListItemDetailMetadataTagList), + title: string(), + tags: array(ItemDetailMetadataTagListItem) +}) +export type ItemDetailMetadataTagList = InferOutput + +export const ItemDetailMetadataSeparator = object({ + nodeName: literal(NodeNameEnum.ListItemDetailMetadataSeparator) +}) +export type ItemDetailMetadataSeparator = InferOutput + +export const ItemDetailMetadataItem = union([ + ItemDetailMetadataLabel, + ItemDetailMetadataLink, + ItemDetailMetadataTagList, + ItemDetailMetadataSeparator +]) +export type ItemDetailMetadataItem = InferOutput + +export const ItemDetailMetadata = object({ + nodeName: literal(NodeNameEnum.ListItemDetailMetadata), + items: array(ItemDetailMetadataItem) +}) +export type ItemDetailMetadata = InferOutput + +export const ItemDetail = object({ + // nodeName: NodeName, + nodeName: literal(NodeNameEnum.ListItemDetail), + children: array(union([Markdown, ItemDetailMetadata])), + width: optional(number()) +}) +export type ItemDetail = InferOutput + +export const Item = object({ + nodeName: literal(NodeNameEnum.ListItem), + title: string(), + subTitle: optional(string()), + accessories: optional(array(ItemAccessory)), + value: string(), + defaultAction: optional(string()), + actions: optional(ActionSchema.ActionPanel), + icon: optional(Icon), + keywords: optional(array(string())) + // id: optional(string('')), +}) +export type Item = InferOutput + +export const Section = object({ + nodeName: literal(NodeNameEnum.ListSection), + title: optional(string()), + subtitle: optional(string()), + items: array(Item) +}) +export type Section = InferOutput + +export const ListInheritOptions = union([ + literal("items"), + literal("detail"), + literal("filter"), + literal("sections"), + literal("actions"), + literal("defaultAction") +]) +export type ListInheritOptions = InferOutput +export const List = object({ + nodeName: literal(NodeNameEnum.List), + sections: optional(array(Section)), + items: optional(array(Item)), + filter: union([literal("none"), literal("default")]), + detail: optional(ItemDetail), + actions: optional(ActionSchema.ActionPanel), + defaultAction: optional(string()), + inherits: optional(array(ListInheritOptions)) +}) +// export type List = InferOutput +/** + * Manually define type of List to avoid TypeScript error + * `Type instantiation is excessively deep and possibly infinite.` + * This happens when this type is used in another package. + * This doesn't seem to happen when this package is published as TypeScript, + * but only when it's packaged and published as JavaScript. + */ +export type List = { + nodeName: NodeNameEnum.List + sections?: Section[] + items?: Item[] + filter: "none" | "default" + detail?: ItemDetail + actions?: ActionSchema.ActionPanel + defaultAction?: string + inherits?: ListInheritOptions[] +} diff --git a/packages/api/src/ui/worker/schema/markdown.ts b/packages/api/src/ui/worker/schema/markdown.ts new file mode 100644 index 0000000..ac827f5 --- /dev/null +++ b/packages/api/src/ui/worker/schema/markdown.ts @@ -0,0 +1,8 @@ +import { object, string, type InferOutput } from "valibot" +import { NodeName, NodeNameEnum } from "../../../models/constants" + +export const Markdown = object({ + nodeName: NodeName, + content: string() +}) +export type Markdown = InferOutput diff --git a/packages/api/src/utils/__tests__/path.test.ts b/packages/api/src/utils/__tests__/path.test.ts new file mode 100644 index 0000000..6c374c0 --- /dev/null +++ b/packages/api/src/utils/__tests__/path.test.ts @@ -0,0 +1,7 @@ +import { describe, expect, test } from "bun:test" +import { translateScopeToPath } from "../path" + +test("translateScopeToPath", async () => { + // const path = await translateScopeToPath("$EXTENSION/foo.txt", "abc") + // expect(path).toBe("abc/foo.txt") +}) diff --git a/packages/api/src/utils/path.ts b/packages/api/src/utils/path.ts new file mode 100644 index 0000000..ca27edf --- /dev/null +++ b/packages/api/src/utils/path.ts @@ -0,0 +1,191 @@ +import * as pathAPI from "@tauri-apps/api/path" +import { BaseDirectory } from "@tauri-apps/api/path" +import { exists, mkdir } from "@tauri-apps/plugin-fs" +import { minimatch } from "minimatch" +import type { + FsPermissionScoped, + OpenPermissionScoped, + ShellPermissionScoped +} from "../permissions/schema" + +export async function combinePathAndBaseDir(target: string, baseDir?: BaseDirectory) { + if (!baseDir) return target + switch (baseDir) { + case BaseDirectory.Desktop: + return await pathAPI.join(await pathAPI.desktopDir(), target) + case BaseDirectory.Document: + return await pathAPI.join(await pathAPI.documentDir(), target) + case BaseDirectory.Download: + return await pathAPI.join(await pathAPI.downloadDir(), target) + case BaseDirectory.Home: + return await pathAPI.join(await pathAPI.homeDir(), target) + case BaseDirectory.AppData: + return await pathAPI.join(await pathAPI.appDataDir(), target) + default: + break + } +} + +export const mapDirAliasToDirFn: Record Promise> = { + $DESKTOP: pathAPI.desktopDir, + $DOCUMENT: pathAPI.documentDir, + $DOWNLOAD: pathAPI.downloadDir, + $HOME: pathAPI.homeDir, + $APPDATA: pathAPI.appDataDir +} + +export const AllPathAliases = [ + ...Object.keys(mapDirAliasToDirFn), + "$EXTENSION", + "$EXTENSION_SUPPORT" +] + +export function pathStartsWithAlias(path: string) { + return AllPathAliases.some((alias) => path.startsWith(alias)) +} + +/** + * @example + * Translate $DESKTOP/* to /Users/username/Desktop/* + * Translate $DOWNLOAD/** to /Users/username/Downloads/** + * @param scope expected to be like $DESKTOP/*, $DOWNLOAD/**, $DOCUMENT/abc/*.txt + */ +export async function translateScopeToPath(scope: string, extensionDir: string): Promise { + if (scope.startsWith("$EXTENSION")) { + return pathAPI.join(extensionDir, scope.slice("$EXTENSION".length)) + } + if (scope.startsWith("$EXTENSION_SUPPORT")) { + const appDataDir = await pathAPI.appDataDir() + if (extensionDir.startsWith(appDataDir)) { + return pathAPI.join(appDataDir, "extensions_support", await pathAPI.basename(extensionDir)) + } else { + const extSupportDir = await pathAPI.join(extensionDir, "extensions_support") + if (!(await exists(extSupportDir))) { + await mkdir(extSupportDir, { recursive: true }) + } + return extSupportDir + } + } + for (const key of Object.keys(mapDirAliasToDirFn)) { + if (scope.startsWith(key)) { + const alias = key + const pattern = scope.slice(key.length) + const dirFn = mapDirAliasToDirFn[alias] + if (!dirFn) { + throw new Error(`Invalid scope alias: ${alias}`) + } + const fullDir = await dirFn() + return pathAPI.join(fullDir, pattern) + } + } + return scope + throw new Error(`Invalid scope: ${scope}`) +} + +/** + * TODO: but this function also does security check to prevent parent directory traversal + * @param target full path to file + * @param scope expected to be like $DESKTOP/*, $DOWNLOAD/** + */ +export async function matchPathAndScope( + target: string, + scope: string, + extensionDir: string +): Promise { + const translatedTarget = await translateScopeToPath(target, extensionDir) + const translatedScope = await translateScopeToPath(scope, extensionDir) + return minimatch(translatedTarget, translatedScope) +} + +/** + * This is a helper function to verify scoped permission for path + * If a scoped permission needs access to paths, this function verify whether the path is allowed by the permission + * @param requiredPermissions + * @param userPermissionScopes + * @param path + * @param options + * @returns + */ +export async function verifyGeneralPathScopedPermission( + requiredPermissions: T, + userPermissionScopes: (FsPermissionScoped | OpenPermissionScoped | ShellPermissionScoped)[], + path: string | URL, + extensionDir: string, + options?: { baseDir?: BaseDirectory } +) { + path = path.toString() + const fullPath = await combinePathAndBaseDir(path, options?.baseDir) + if (!fullPath) { + throw new Error("Invalid path or base directory") + } + const matchedPermissionScope = userPermissionScopes.filter((p) => + requiredPermissions.includes(p.permission) + ) + if (matchedPermissionScope.length === 0) { + throw new Error( + `Path Permission denied. Require one of these: [${requiredPermissions.join(", ")}] for path: ${fullPath}` + ) + } + + for (const permission of matchedPermissionScope) { + // deny has priority, if deny rule is matched, we ignore allow rule + for (const deny of permission.deny || []) { + if (!deny.path) continue + if (await matchPathAndScope(fullPath, deny.path, extensionDir)) { + throw new Error(`Permission denied for path: ${fullPath} by rule ${deny.path}`) + } + } + for (const allow of permission.allow || []) { + if (!allow.path) continue + if (await matchPathAndScope(fullPath, allow.path, extensionDir)) { + return + } + } + } + // No Allow rule and path matched + throw new Error(`Permission denied for path: ${path}, no rule matched.`) +} + +/** + * This permission verifier helps to verify scoped permission for URL or path + * Pass in user scoped permission, value and key to verify, return true if permission is allowed + * @param userPermissionScopes + * @param value + * @param key + * @returns + */ +export async function verifyScopedPermission( + userPermissionScopes: (FsPermissionScoped | OpenPermissionScoped | ShellPermissionScoped)[], + value: string, + key: "url" | "path", + extensionDir: string +): Promise { + async function match(value: string, scope: string): Promise { + if (key === "url") { + return minimatch(value, scope) + } else if (key === "path") { + return matchPathAndScope(value, scope, extensionDir) + } else { + throw new Error(`Invalid key: ${key}`) + } + } + let pass = false + for (const permission of userPermissionScopes) { + for (const allow of permission.allow || []) { + if (allow[key] && (await match(value, allow[key]))) { + pass = true + break + } + } + if (pass) { + break + } + for (const deny of permission.deny || []) { + if (deny[key] && (await match(value, deny[key]))) { + pass = false + break + } + } + } + return pass +} diff --git a/packages/api/src/utils/permission-check.ts b/packages/api/src/utils/permission-check.ts new file mode 100644 index 0000000..5313857 --- /dev/null +++ b/packages/api/src/utils/permission-check.ts @@ -0,0 +1,14 @@ +/** + * The simplest permission check function, will throw error if the user does not have the required permission instead of returning false. + * Both parameters are expected to be string arrays. We use generic to make sure the type is correct. + * @param userPermissions + * @param requiredPermissions + */ +export function checkPermission( + userPermissions: Permission[], + requiredPermissions: Permission[] +) { + if (!requiredPermissions.some((p) => userPermissions.includes(p))) { + throw new Error(`${requiredPermissions.join(" or ")} permission is required`) + } +} diff --git a/packages/api/src/version.ts b/packages/api/src/version.ts new file mode 100644 index 0000000..ec7bba0 --- /dev/null +++ b/packages/api/src/version.ts @@ -0,0 +1,32 @@ +import { clean, gte, parse, satisfies, sort } from "semver" +import * as v from "valibot" + +export const breakingChangesVersionCheckpoints = [ + { version: "0.0.1", changelog: "" }, + { version: "0.0.12-beta.0", changelog: "" }, + { version: "0.0.12-beta.1", changelog: "" }, + { + version: "0.0.17", + changelog: "New Custom UI loading method, remove base url config requirement." + } +] +const checkpointVersions = breakingChangesVersionCheckpoints.map((c) => c.version) +const sortedCheckpointVersions = sort(checkpointVersions) + +export const version = "0.0.27" + +export function isVersionBetween(v: string, start: string, end: string) { + const vCleaned = clean(v) + const startCleaned = clean(start) + const endCleaned = clean(end) + return satisfies(vCleaned!, `${startCleaned} - ${endCleaned}`) +} + +/** + * Check if a given version is compatible with this API version + * Basically, this checks if the version is greater than or equal to the last checkpoint version + * @param version + */ +export function isCompatible(version: string) { + return gte(version, v.parse(v.string(), sortedCheckpointVersions.at(-1))) +} diff --git a/packages/api/tsconfig.json b/packages/api/tsconfig.json new file mode 100644 index 0000000..bb9893e --- /dev/null +++ b/packages/api/tsconfig.json @@ -0,0 +1,28 @@ +{ + "extends": "../typescript-config/base.json", + "compilerOptions": { + // Enable latest features + "lib": ["ESNext", "DOM"], + "target": "ESNext", + "module": "ESNext", + "moduleDetection": "force", + "jsx": "react-jsx", + "allowJs": true, + + // Bundler mode + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "noEmit": true, + + // Best practices + "strict": true, + "skipLibCheck": true, + "noFallthroughCasesInSwitch": true, + + // Some stricter flags (disabled by default) + "noUnusedLocals": false, + "noUnusedParameters": false, + "noPropertyAccessFromIndexSignature": false + } +} diff --git a/packages/api/typedoc.json b/packages/api/typedoc.json new file mode 100644 index 0000000..cc45e70 --- /dev/null +++ b/packages/api/typedoc.json @@ -0,0 +1,20 @@ +{ + "$schema": "https://typedoc.org/schema.json", + "entryPoints": [ + "src/index.ts", + "src/ui/index.ts", + "src/ui/iframe/index.ts", + "src/ui/worker/index.ts", + "src/models/index.ts", + "src/commands/index.ts", + "src/ui/api/permissions.ts", + "src/supabase/index.ts" + ], + "visibilityFilters": { + "protected": false, + "private": false, + "inherited": true, + "external": true + }, + "out": "docs" +} diff --git a/packages/ci/.gitignore b/packages/ci/.gitignore new file mode 100644 index 0000000..9b1ee42 --- /dev/null +++ b/packages/ci/.gitignore @@ -0,0 +1,175 @@ +# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs + +logs +_.log +npm-debug.log_ +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Caches + +.cache + +# Diagnostic reports (https://nodejs.org/api/report.html) + +report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json + +# Runtime data + +pids +_.pid +_.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover + +lib-cov + +# Coverage directory used by tools like istanbul + +coverage +*.lcov + +# nyc test coverage + +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) + +.grunt + +# Bower dependency directory (https://bower.io/) + +bower_components + +# node-waf configuration + +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) + +build/Release + +# Dependency directories + +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) + +web_modules/ + +# TypeScript cache + +*.tsbuildinfo + +# Optional npm cache directory + +.npm + +# Optional eslint cache + +.eslintcache + +# Optional stylelint cache + +.stylelintcache + +# Microbundle cache + +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history + +.node_repl_history + +# Output of 'npm pack' + +*.tgz + +# Yarn Integrity file + +.yarn-integrity + +# dotenv environment variable files + +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) + +.parcel-cache + +# Next.js build output + +.next +out + +# Nuxt.js build / generate output + +.nuxt +dist + +# Gatsby files + +# Comment in the public line in if your project uses Gatsby and not Next.js + +# https://nextjs.org/blog/next-9-1#public-directory-support + +# public + +# vuepress build output + +.vuepress/dist + +# vuepress v2.x temp and cache directory + +.temp + +# Docusaurus cache and generated files + +.docusaurus + +# Serverless directories + +.serverless/ + +# FuseBox cache + +.fusebox/ + +# DynamoDB Local files + +.dynamodb/ + +# TernJS port file + +.tern-port + +# Stores VSCode versions used for testing VSCode extensions + +.vscode-test + +# yarn v2 + +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# IntelliJ based IDEs +.idea + +# Finder (MacOS) folder config +.DS_Store diff --git a/packages/ci/README.md b/packages/ci/README.md new file mode 100644 index 0000000..0317eb3 --- /dev/null +++ b/packages/ci/README.md @@ -0,0 +1 @@ +This package contains scripts for CI/CD, also exported as a library for other packages to use. diff --git a/packages/ci/index.ts b/packages/ci/index.ts new file mode 100644 index 0000000..4a96093 --- /dev/null +++ b/packages/ci/index.ts @@ -0,0 +1 @@ +export * from "./src/path.ts" diff --git a/packages/ci/package.json b/packages/ci/package.json new file mode 100644 index 0000000..3ad3c7d --- /dev/null +++ b/packages/ci/package.json @@ -0,0 +1,15 @@ +{ + "name": "@kksh/ci", + "module": "index.ts", + "type": "module", + "scripts": { + "prepare": "bun scripts/setup.ts" + }, + "devDependencies": { + "@types/bun": "latest", + "@kksh/typescript-config": "workspace:*" + }, + "peerDependencies": { + "typescript": "^5.0.0" + } +} diff --git a/packages/ci/scripts/init-env.ts b/packages/ci/scripts/init-env.ts new file mode 100644 index 0000000..5cffc12 --- /dev/null +++ b/packages/ci/scripts/init-env.ts @@ -0,0 +1,54 @@ +/** + * Initialize .env files for packages + * All env variables here are public + */ +import { writeFileSync } from "fs" +import { join } from "path" +import { REPO_ROOT } from "@/path" + +console.log("Init Env") + +const defaultEnvUrl = `https://storage.kunkun.sh/env.json` +const res = await fetch(defaultEnvUrl) +const env = await res.json() + +let envContent = "" + +if (!process.env.SUPABASE_ANON_KEY) { + process.env.SUPABASE_ANON_KEY = env.SUPABASE_ANON_KEY +} +if (!process.env.SUPABASE_PROJECT_ID) { + process.env.SUPABASE_PROJECT_ID = env.SUPABASE_PROJECT_ID +} + +if (process.env.SUPABASE_ANON_KEY) { + envContent += `SUPABASE_ANON_KEY=${process.env.SUPABASE_ANON_KEY}\n` +} +if (process.env.SUPABASE_PROJECT_ID) { + const supabaseUrl = `https://${process.env.SUPABASE_PROJECT_ID}.supabase.co` + const supabaseGraphqlEndpoint = `${supabaseUrl}/graphql/v1` + envContent += ` +SUPABASE_GRAPHQL_ENDPOINT=${supabaseGraphqlEndpoint} +SUPABASE_URL=${supabaseUrl} +` +} +if (process.env.SUPABASE_SERVICE_ROLE_KEY) { + envContent += `\nSUPABASE_SERVICE_ROLE_KEY=${process.env.SUPABASE_SERVICE_ROLE_KEY}\n` +} +if (process.env.POSTHOG_PUBLIC_KEY && process.env.POSTHOG_HOST) { + envContent += ` +POSTHOG_PUBLIC_KEY=${process.env.POSTHOG_PUBLIC_KEY} +POSTHOG_HOST=${process.env.POSTHOG_HOST} +` +} + +// writeFileSync(join(__dirname, "../apps/desktop/.env"), envContent) +writeFileSync( + join(REPO_ROOT, "apps/desktop/.env"), + ` +PUBLIC_SUPABASE_ANON_KEY=${process.env.SUPABASE_ANON_KEY} +PUBLIC_SUPABASE_PROJECT_ID=${process.env.SUPABASE_PROJECT_ID} +` +) +// writeFileSync(join(__dirname, "../packages/gql/.env"), envContent) +writeFileSync(join(REPO_ROOT, "packages/schema/.env"), envContent) diff --git a/packages/ci/scripts/setup.ts b/packages/ci/scripts/setup.ts new file mode 100644 index 0000000..81cbbdd --- /dev/null +++ b/packages/ci/scripts/setup.ts @@ -0,0 +1,6 @@ +import path from "path" +import { PACKAGES_PATHS } from "@/path" +import { $ } from "bun" + +// Initialize .env files +await $`bun ${path.join(PACKAGES_PATHS.CI, "scripts", "init-env.ts")}` diff --git a/packages/ci/src/path.ts b/packages/ci/src/path.ts new file mode 100644 index 0000000..a4f2f6e --- /dev/null +++ b/packages/ci/src/path.ts @@ -0,0 +1,12 @@ +import path from "path" +import { fileURLToPath } from "url" + +const filepath = fileURLToPath(import.meta.url) +export const REPO_ROOT = path.dirname(path.dirname(path.dirname(path.dirname(filepath)))) +export const PACKAGES_PATHS = { + CI: path.join(REPO_ROOT, "packages/ci"), + API: path.join(REPO_ROOT, "packages/api"), + SCHEMA: path.join(REPO_ROOT, "packages/schema"), + EXTENSIONS: path.join(REPO_ROOT, "packages/extensions"), + TEMPLATES: path.join(REPO_ROOT, "packages/templates") +} diff --git a/packages/ci/tsconfig.json b/packages/ci/tsconfig.json new file mode 100644 index 0000000..221a97b --- /dev/null +++ b/packages/ci/tsconfig.json @@ -0,0 +1,9 @@ +{ + "extends": "../typescript-config/base.json", + "compilerOptions": { + "baseUrl": ".", + "paths": { + "@/*": ["src/*"] + } + } +} diff --git a/packages/db/Cargo.toml b/packages/db/Cargo.toml new file mode 100644 index 0000000..580ec6a --- /dev/null +++ b/packages/db/Cargo.toml @@ -0,0 +1,16 @@ +[package] +name = "db" +version = "0.1.0" +edition = "2021" + +[dependencies] +rusqlite = { version = "0.31.0", features = [ + "bundled", + # "bundled-sqlcipher" +] } +serde = { workspace = true, features = ["derive"] } +serde_json = { workspace = true } +tempfile = "3.10.1" +strum = { workspace = true } +strum_macros = { workspace = true } +anyhow.workspace = true diff --git a/packages/db/sql/2024-10-23.sql b/packages/db/sql/2024-10-23.sql new file mode 100644 index 0000000..5da287f --- /dev/null +++ b/packages/db/sql/2024-10-23.sql @@ -0,0 +1,96 @@ +CREATE TABLE IF NOT EXISTS schema_version (version INTEGER NOT NULL); + +-- Extensions table +CREATE TABLE IF NOT EXISTS extensions ( + ext_id INTEGER PRIMARY KEY AUTOINCREMENT, + identifier TEXT NOT NULL, + version TEXT NOT NULL, + enabled BOOLEAN DEFAULT TRUE, + path TEXT UNIQUE, + data JSON, + installed_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP +); + +CREATE TABLE IF NOT EXISTS commands ( + cmd_id INTEGER PRIMARY KEY AUTOINCREMENT, + ext_id INTEGER NOT NULL, + name TEXT NOT NULL, + enabled BOOLEAN DEFAULT TRUE, + alias TEXT, + hotkey TEXT, + type TEXT NOT NULL CHECK ( + type IN ( + 'iframe', + 'ui_worker', + 'headless_worker', + 'quick_link', + 'remote' + ) + ), + data JSON, + FOREIGN KEY (ext_id) REFERENCES extensions (ext_id) +); + +-- Extension Data table +CREATE TABLE IF NOT EXISTS extension_data ( + data_id INTEGER PRIMARY KEY AUTOINCREMENT, + ext_id INTEGER NOT NULL, + data_type TEXT NOT NULL, + data JSON NOT NULL, + search_text TEXT, + created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, + updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, + FOREIGN KEY (ext_id) REFERENCES extensions (ext_id) +); + +-- Full-text search index for ext_data +CREATE VIRTUAL TABLE IF NOT EXISTS extension_data_fts USING fts5 ( + data_id UNINDEXED, + search_text, + content = extension_data, + content_rowid = data_id +); + +-- Trigger to update FTS index when extension_data is inserted +CREATE TRIGGER IF NOT EXISTS extension_data_ai AFTER INSERT ON extension_data BEGIN +INSERT INTO + extension_data_fts (data_id, search_text) +VALUES + (new.data_id, new.search_text); + +END; + +-- Trigger to update FTS index when extension_data is updated +CREATE TRIGGER IF NOT EXISTS extension_data_au AFTER +UPDATE ON extension_data BEGIN +INSERT INTO + extension_data_fts (extension_data_fts, data_id, search_text) +VALUES + ('delete', old.data_id, old.search_text); + +INSERT INTO + extension_data_fts (data_id, search_text) +VALUES + (new.data_id, new.search_text); + +END; + +-- Trigger to update FTS index when extension_data is deleted +CREATE TRIGGER IF NOT EXISTS extension_data_ad AFTER DELETE ON extension_data BEGIN +INSERT INTO + extension_data_fts (extension_data_fts, data_id, search_text) +VALUES + ('delete', old.data_id, old.search_text); + +END; + +-- Trigger to update 'updated_at' timestamp when extension data is updated +CREATE TRIGGER IF NOT EXISTS update_extension_data_timestamp AFTER +UPDATE ON extension_data BEGIN +UPDATE extension_data +SET + updated_at = CURRENT_TIMESTAMP +WHERE + data_id = NEW.data_id; + +END; diff --git a/packages/db/src/lib.rs b/packages/db/src/lib.rs new file mode 100644 index 0000000..c9d2ac9 --- /dev/null +++ b/packages/db/src/lib.rs @@ -0,0 +1,888 @@ +pub mod models; +pub mod schema; +use models::CmdType; +use rusqlite::{params, params_from_iter, Connection, Error, Result, ToSql}; +use serde::{Deserialize, Serialize}; +use std::path::{self, Path}; +use strum_macros::{Display, EnumString}; + +pub const DB_VERSION: u32 = 1; + +pub fn get_connection>( + file_path: P, + encryption_key: Option, +) -> Result { + let conn = Connection::open(file_path)?; + if let Some(encryption_key) = encryption_key { + conn.pragma_update(None, "key", &encryption_key)?; + } + Ok(conn) +} + +#[derive(Debug)] +pub struct JarvisDB { + pub conn: Connection, +} + +#[derive(Debug, Serialize, Deserialize, Display)] +#[serde(rename_all = "UPPERCASE")] +pub enum SQLSortOrder { + Asc, + Desc, +} + +#[derive(Debug, Serialize, Deserialize, Display, PartialEq)] +#[serde(rename_all = "snake_case")] +pub enum ExtDataField { + Data, + SearchText, +} + +impl JarvisDB { + pub fn new>(file_path: P, encryption_key: Option) -> Result { + let conn = get_connection(file_path, encryption_key)?; + Ok(Self { conn }) + } + + /** + * Run this when app starts everytime, to ensure the db is up to date. + */ + pub fn init(&self) -> Result<()> { + let schema_version_exists = self.schema_version_exists()?; + if !schema_version_exists { + // this means the db is not initialized at all, so we need to migrate from version 0, i.e. run all migration scripts + self.migrate_after_version(0)?; + } else { + let current_version = self.get_schema_version()?; + self.migrate_after_version(current_version.unwrap())?; + } + Ok(()) + } + + pub fn migrate_after_version(&self, version: u16) -> Result<()> { + for migration in schema::MIGRATIONS.iter() { + if migration.version > version { + println!( + "Migrating from version {} to {}", + version, migration.version + ); + // self.conn.execute(&migration.schema, params![])?; + match self + .conn + .execute_batch(&format!("BEGIN; {} COMMIT;", migration.script)) + { + Ok(_) => { + self.upsert_schema_version(migration.version)?; + } + Err(e) => { + eprintln!("Failed to execute migration script: {}", e); + return Err(e); + } + } + } + } + Ok(()) + } + + /** + * Insert the schema version into the schema_version table if it doesn't exist yet + */ + pub fn upsert_schema_version(&self, version: u16) -> Result<()> { + self.conn.execute( + "INSERT OR IGNORE INTO schema_version (version) VALUES (?1)", + params![version], + )?; + Ok(()) + } + + pub fn schema_version_exists(&self) -> Result { + match self.get_schema_version() { + Ok(Some(_)) => Ok(true), + _ => Ok(false), + } + } + + pub fn get_schema_version(&self) -> Result> { + let mut stmt = self + .conn + .prepare("SELECT version FROM schema_version ORDER BY version DESC LIMIT 1;")?; + let version_iter = stmt.query_map(params![], |row| { + let version: u16 = row.get(0)?; + Ok(version) + })?; + let mut versions: Vec = Vec::new(); + for version in version_iter { + versions.push(version?); + } + Ok(versions.first().copied()) + } + + /* -------------------------------------------------------------------------- */ + /* Extensions CRUD */ + /* -------------------------------------------------------------------------- */ + + pub fn create_extension( + &self, + identifier: &str, + version: &str, + enabled: bool, + path: Option<&str>, + data: Option<&str>, + ) -> Result<()> { + self.conn.execute( + "INSERT INTO extensions (identifier, version, enabled, path, data) VALUES (?1, ?2, ?3, ?4, ?5)", + params![identifier, version, enabled, path, data], + )?; + Ok(()) + } + + pub fn get_all_extensions(&self) -> Result> { + let mut stmt = self.conn.prepare( + "SELECT ext_id, identifier, path, data, version, enabled, installed_at FROM extensions", + )?; + let ext_iter = stmt.query_map(params![], |row| { + Ok(models::Ext { + ext_id: row.get(0)?, + identifier: row.get(1)?, + path: row.get(2)?, + data: row.get(3)?, + version: row.get(4)?, + enabled: row.get(5)?, + installed_at: row.get(6)?, + }) + })?; + let mut exts = Vec::new(); + for ext in ext_iter { + exts.push(ext?); + } + Ok(exts) + } + + /** + * Get the first extension by identifier, if there are multiple, it will return an error + */ + pub fn get_unique_extension_by_identifier( + &self, + identifier: &str, + ) -> anyhow::Result> { + let exts = self.get_all_extensions_by_identifier(identifier)?; + if exts.len() > 1 { + return Err(anyhow::anyhow!( + "Multiple extensions with the same identifier: {}", + identifier + )); + } + Ok(exts.first().cloned()) + } + + pub fn get_all_extensions_by_identifier(&self, identifier: &str) -> Result> { + let mut stmt = self.conn.prepare( + "SELECT ext_id, identifier, path, data, version, enabled, installed_at FROM extensions WHERE identifier = ?1", + )?; + let ext_iter = stmt.query_map(params![identifier], |row| { + Ok(models::Ext { + ext_id: row.get(0)?, + identifier: row.get(1)?, + path: row.get(2)?, + data: row.get(3)?, + version: row.get(4)?, + enabled: row.get(5)?, + installed_at: row.get(6)?, + }) + })?; + let mut exts = Vec::new(); + for ext in ext_iter { + exts.push(ext?); + } + Ok(exts) + } + + pub fn get_unique_extension_by_path(&self, path: &str) -> Result> { + let mut stmt = self.conn.prepare( + "SELECT ext_id, identifier, path, data, version, enabled, installed_at FROM extensions WHERE path = ?1", + )?; + let ext_iter = stmt.query_map(params![path], |row| { + Ok(models::Ext { + ext_id: row.get(0)?, + identifier: row.get(1)?, + path: row.get(2)?, + data: row.get(3)?, + version: row.get(4)?, + enabled: row.get(5)?, + installed_at: row.get(6)?, + }) + })?; + let mut exts = Vec::new(); + for ext in ext_iter { + exts.push(ext?); + } + Ok(exts.first().cloned()) + } + + // TODO: clean this up + // pub fn delete_extension_by_identifier(&self, identifier: &str) -> Result<()> { + // self.conn.execute( + // "DELETE FROM extensions WHERE identifier = ?1", + // params![identifier], + // )?; + // Ok(()) + // } + + pub fn delete_extension_by_path(&self, path: &str) -> Result<()> { + self.conn + .execute("DELETE FROM extensions WHERE path = ?1", params![path])?; + Ok(()) + } + + pub fn delete_extension_by_ext_id(&self, ext_id: i32) -> Result<()> { + self.conn + .execute("DELETE FROM extensions WHERE ext_id = ?1", params![ext_id])?; + Ok(()) + } + + /* -------------------------------------------------------------------------- */ + /* Command CRUD */ + /* -------------------------------------------------------------------------- */ + pub fn create_command( + &self, + ext_id: i32, + name: &str, + cmd_type: CmdType, + data: &str, + enabled: bool, + alias: Option<&str>, + hotkey: Option<&str>, + ) -> Result<()> { + self.conn.execute( + "INSERT INTO commands (ext_id, name, type, data, alias, hotkey, enabled) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7)", + params![ext_id, name, cmd_type.to_string(), data, alias, hotkey, enabled], + )?; + Ok(()) + } + + pub fn get_command_by_id(&self, cmd_id: i32) -> Result> { + let mut stmt = self + .conn + .prepare("SELECT cmd_id, ext_id, name, type, data, alias, hotkey, enabled FROM commands WHERE cmd_id = ?1")?; + let cmd_iter = stmt.query_map(params![cmd_id], |row| { + Ok(models::Cmd { + cmd_id: row.get(0)?, + ext_id: row.get(1)?, + name: row.get(2)?, + type_: row.get(3)?, + data: row.get(4)?, + alias: row.get(5)?, + hotkey: row.get(6)?, + enabled: row.get(7)?, + }) + })?; + let mut cmds = Vec::new(); + for cmd in cmd_iter { + cmds.push(cmd?); + } + Ok(cmds.first().cloned()) + } + + pub fn get_commands_by_ext_id(&self, ext_id: i32) -> Result> { + let mut stmt = self + .conn + .prepare("SELECT cmd_id, ext_id, name, type, data, alias, hotkey, enabled FROM commands WHERE ext_id = ?1")?; + let cmd_iter = stmt.query_map(params![ext_id], |row| { + Ok(models::Cmd { + cmd_id: row.get(0)?, + ext_id: row.get(1)?, + name: row.get(2)?, + type_: row.get(3)?, + data: row.get(4)?, + alias: row.get(5)?, + hotkey: row.get(6)?, + enabled: row.get(7)?, + }) + })?; + let mut cmds = Vec::new(); + for cmd in cmd_iter { + cmds.push(cmd?); + } + Ok(cmds) + } + + pub fn delete_command_by_id(&self, cmd_id: i32) -> Result<()> { + self.conn + .execute("DELETE FROM commands WHERE cmd_id = ?1", params![cmd_id])?; + Ok(()) + } + + pub fn update_command_by_id( + &self, + cmd_id: i32, + name: &str, + cmd_type: CmdType, + data: &str, + enabled: bool, + alias: Option<&str>, + hotkey: Option<&str>, + ) -> Result<()> { + self.conn.execute( + "UPDATE commands SET name = ?1, type = ?2, data = ?3, alias = ?4, hotkey = ?5, enabled = ?6 WHERE cmd_id = ?7", + params![name, cmd_type.to_string(), data, alias, hotkey, enabled, cmd_id], + )?; + Ok(()) + } + + /* -------------------------------------------------------------------------- */ + /* Extension Data CRUD */ + /* -------------------------------------------------------------------------- */ + pub fn create_extension_data( + &self, + ext_id: i32, + data_type: &str, + data: &str, + search_text: Option<&str>, + ) -> Result<()> { + self.conn.execute( + "INSERT INTO extension_data (ext_id, data_type, data, search_text) VALUES (?1, ?2, ?3, ?4)", + params![ext_id, data_type, data, search_text], + )?; + Ok(()) + } + + pub fn get_extension_data_by_id(&self, data_id: i32) -> Result> { + let mut stmt = self.conn.prepare( + "SELECT data_id, ext_id, data_type, data, search_text, created_at, updated_at FROM extension_data WHERE data_id = ?1", + )?; + let ext_data_iter = stmt.query_map(params![data_id], |row| { + Ok(models::ExtData { + data_id: row.get(0)?, + ext_id: row.get(1)?, + data_type: row.get(2)?, + data: row.get(3)?, + search_text: row.get(4)?, + created_at: row.get(5)?, + updated_at: row.get(6)?, + }) + })?; + let mut ext_data = Vec::new(); + for data in ext_data_iter { + ext_data.push(data?); + } + Ok(ext_data.first().cloned()) + } + + pub fn search_extension_data( + &self, + ext_id: i32, + search_exact_match: bool, + data_id: Option, + data_type: Option<&str>, + search_text: Option<&str>, + after_created_at: Option<&str>, + before_created_at: Option<&str>, + limit: Option, + order_by_created_at: Option, + order_by_updated_at: Option, + fields: Option>, + ) -> Result> { + let mut fields = fields; + if fields.is_none() { + fields = Some(vec![ExtDataField::Data, ExtDataField::SearchText]); + } + let contains_data_field = fields.as_ref().map_or(false, |fields| { + fields.iter().any(|f| f == &ExtDataField::Data) + }); + let contains_search_text_field = fields.as_ref().map_or(false, |fields| { + fields.iter().any(|f| f == &ExtDataField::SearchText) + }); + let mut query = String::from("SELECT data_id, ext_id, data_type, created_at, updated_at"); + if contains_data_field { + query.push_str(", data"); + } + if contains_search_text_field { + query.push_str(", search_text"); + } + query.push_str( + " FROM extension_data + WHERE ext_id = ?1", + ); + let mut params: Vec> = vec![Box::new(ext_id)]; + let mut param_index = 2; + + if let Some(di) = data_id { + query.push_str(&format!(" AND data_id = ?{}", param_index)); + params.push(Box::new(di)); + param_index += 1; + } + + if let Some(dt) = data_type { + query.push_str(&format!(" AND data_type = ?{}", param_index)); + params.push(Box::new(dt)); + param_index += 1; + } + + if search_exact_match { + if let Some(st) = search_text { + query.push_str(&format!(" AND search_text = ?{}", param_index)); + params.push(Box::new(st)); + param_index += 1; + } + } else { + if let Some(st) = search_text { + query.push_str(&format!(" AND search_text LIKE ?{}", param_index)); + params.push(Box::new(format!("%{}%", st))); + param_index += 1; + } + } + + if let Some(after) = after_created_at { + query.push_str(&format!(" AND created_at > ?{}", param_index)); + params.push(Box::new(after)); + param_index += 1; + } + + if let Some(before) = before_created_at { + query.push_str(&format!(" AND created_at < ?{}", param_index)); + params.push(Box::new(before)); + param_index += 1; + } + + if let Some(order_by_created_at) = order_by_created_at { + query.push_str(&format!( + " ORDER BY created_at {}", + order_by_created_at.to_string() + )); + } + + if let Some(order_by_updated_at) = order_by_updated_at { + query.push_str(&format!( + " ORDER BY updated_at {}", + order_by_updated_at.to_string() + )); + } + + if let Some(limit) = limit { + query.push_str(&format!(" LIMIT ?{}", param_index)); + params.push(Box::new(limit)); + } + let mut stmt = self.conn.prepare(&query)?; + + let ext_data_iter = + stmt.query_map(params_from_iter(params.iter().map(|p| p.as_ref())), |row| { + Ok(models::ExtData { + data_id: row.get(0)?, + ext_id: row.get(1)?, + data_type: row.get(2)?, + created_at: row.get(3)?, + updated_at: row.get(4)?, + data: match contains_data_field { + true => Some(row.get(5)?), + false => None, + }, + search_text: match contains_search_text_field { + true => row.get(5 + contains_data_field as usize)?, // if contains_data_field is true, search_text is at index 6, otherwise 5 + false => None, + }, + }) + })?; + + let mut ext_data = Vec::new(); + for data in ext_data_iter { + ext_data.push(data?); + } + Ok(ext_data) + } + + pub fn delete_extension_data_by_id(&self, data_id: i32) -> Result<()> { + self.conn.execute( + "DELETE FROM extension_data WHERE data_id = ?1", + params![data_id], + )?; + Ok(()) + } + + pub fn update_extension_data_by_id( + &self, + data_id: i32, + data: &str, + search_text: Option<&str>, + ) -> Result<()> { + self.conn.execute( + "UPDATE extension_data SET data = ?1, search_text = ?2 WHERE data_id = ?3", + params![data, search_text, data_id], + )?; + Ok(()) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use std::{fs, path::PathBuf}; + use tempfile::tempdir; + + #[test] + fn test_get_connection() { + let dir = tempdir().unwrap(); + let db_path = dir.path().join("test.db"); + let _conn = get_connection(&db_path, None).unwrap(); + assert!(fs::metadata(&db_path).is_ok()); + fs::remove_file(&db_path).unwrap(); + } + + #[test] + fn test_extension_crud() { + let dir = tempdir().unwrap(); + let db_path = dir.path().join("test.db"); + // create database and initialize + let db = JarvisDB::new(&db_path, None).unwrap(); + assert!(fs::metadata(&db_path).is_ok()); + db.init().unwrap(); + db.create_extension("test", "0.1.0", true, Some("/abc/def"), None) + .unwrap(); + let exts = db.get_all_extensions().unwrap(); + assert_eq!(exts.len(), 1); + + // expect error due to unique identifier constraint + assert!(db + .create_extension("test", "0.1.0", true, Some("/abc/def"), None) + .is_err()); + + // get ext by identifier + let ext = db.get_unique_extension_by_identifier("test").unwrap(); + assert!(ext.is_some()); + let ext = ext.unwrap(); + assert_eq!(ext.identifier, "test"); + assert_eq!(ext.version, "0.1.0"); + assert_eq!(ext.enabled, true); + assert_eq!(ext.installed_at.len(), 19); + + // get ext by identifier that does not exist + let ext = db.get_unique_extension_by_identifier("test2").unwrap(); + assert!(ext.is_none()); + + /* ----------------------- Delete ext by identifier ---------------------- */ + // db.delete_extension_by_identifier("test").unwrap(); + db.delete_extension_by_path("/abc/def").unwrap(); + let exts = db.get_all_extensions().unwrap(); + assert_eq!(exts.len(), 0); + + fs::remove_file(&db_path).unwrap(); + } + + #[test] + fn test_ext_data_crud() { + // let dir = tempdir().unwrap(); + // let db_path = dir.path().join("test.db"); + let db_path = PathBuf::from("./test.db"); + if db_path.exists() { + fs::remove_file(&db_path).unwrap(); + } + + // create database and initialize + let db = JarvisDB::new(&db_path, None).unwrap(); + assert!(fs::metadata(&db_path).is_ok()); + db.init().unwrap(); + db.create_extension("test", "0.1.0", true, Some("/abc/def"), None) + .unwrap(); + let ext = db + .get_unique_extension_by_identifier("test") + .unwrap() + .unwrap(); + + db.create_extension_data(ext.ext_id, "test", "{}", None) + .unwrap(); + db.create_extension_data(ext.ext_id, "setting", "{}", None) + .unwrap(); + /* ---------------------- Search with data_type == test --------------------- */ + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + Some("test"), + None, + None, + None, + None, + None, + None, + None, + ) + .unwrap(); + + assert_eq!(ext_data.len(), 1); // there is only one record with data_type == test + + /* ------------------------ Search without any filter ----------------------- */ + let ext_data = db + .search_extension_data( + ext.ext_id, false, None, None, None, None, None, None, None, None, None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 2); // one test, one setting + + /* -------------------------- Test Full Text Search ------------------------- */ + db.create_extension_data(ext.ext_id, "data", "{}", Some("hello world from rust")) + .unwrap(); + db.create_extension_data(ext.ext_id, "data", "{}", Some("world is a mess")) + .unwrap(); + /* ----------------------- both record contains world ----------------------- */ + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + Some("data"), + Some("wOrLd"), + None, + None, + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 2); + /* ------------------------ search for rust with FTS ------------------------ */ + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + Some("data"), + Some("rust"), + None, + None, + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 1); + + // get ext data with search text that does not exist + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + Some("test"), + Some("test"), + None, + None, + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 0); + + /* ---------------- All 4 test records are created after 2021 --------------- */ + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + None, + None, + Some("2021-01-01"), + None, + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 4); + + // I don't think this code(or I) could live long enough to see this test fail 2100 + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + None, + None, + Some("2100-01-01"), + None, + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 0); + + /* --------------- All 4 test records are created before 2030 --------------- */ + // if this code still runs in 2030, I will be very happy to fix this test + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + None, + None, + None, + Some("2030-01-01"), + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 4); + + // get ext data with created_at filter that does not exist + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + None, + None, + None, + Some("2021-01-01"), + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 0); + + /* ---------------------- Delete ext data by data_id ---------------------- */ + // there is only one record with data_type == setting + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + Some("setting"), + None, + None, + None, + None, + None, + None, + None, + ) + .unwrap(); + let data_id = ext_data.first().unwrap().data_id; + db.delete_extension_data_by_id(data_id).unwrap(); + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + Some("setting"), + None, + None, + None, + None, + None, + None, + None, + ) + .unwrap(); + assert_eq!(ext_data.len(), 0); + + /* ---------------------- Update ext data by data_id ---------------------- */ + let ext_data = db + .search_extension_data( + ext.ext_id, + false, + None, + Some("data"), + None, + None, + None, + None, + None, + None, + None, + ) + .unwrap(); + let data_id = ext_data.first().unwrap().data_id; + db.update_extension_data_by_id(data_id, "{\"name\": \"huakun\"}", Some("updated")) + .unwrap(); + let ext_data = db.get_extension_data_by_id(data_id).unwrap(); + assert!(ext_data.is_some()); + let ext_data = ext_data.unwrap(); + assert_eq!(ext_data.data.unwrap(), "{\"name\": \"huakun\"}"); + + /* ----------------------------- Optional Fields ---------------------------- */ + // let ext_data = db + // .search_extension_data(ext.ext_id, false, None, Some("data"), None, None, None, None, None, None, vec![]) + // .unwrap(); + + fs::remove_file(&db_path).unwrap(); + } + + #[test] + fn test_command_crud() { + let dir = tempdir().unwrap(); + let db_path = dir.path().join("test.db"); + // create database and initialize + let db = JarvisDB::new(&db_path, None).unwrap(); + assert!(fs::metadata(&db_path).is_ok()); + db.init().unwrap(); + db.create_extension("test", "0.1.0", true, None, None) + .unwrap(); + let ext = db + .get_unique_extension_by_identifier("test") + .unwrap() + .unwrap(); + + db.create_command(ext.ext_id, "test", CmdType::Iframe, "{}", true, None, None) + .unwrap(); + db.create_command( + ext.ext_id, + "test2", + CmdType::UiWorker, + "{}", + true, + Some("t2"), + Some("t2"), + ) + .unwrap(); + /* ---------------------- Get command by id ---------------------- */ + let cmd = db.get_command_by_id(1).unwrap().unwrap(); + assert_eq!(cmd.name, "test"); + assert_eq!(cmd.type_, models::CmdType::Iframe); + assert_eq!(cmd.data, "{}"); + + /* ---------------------- Get commands by ext_id ---------------------- */ + let cmds = db.get_commands_by_ext_id(ext.ext_id).unwrap(); + assert_eq!(cmds.len(), 2); + + // test command test2's alias and hotkey + let cmd = db.get_command_by_id(cmds[1].cmd_id).unwrap().unwrap(); + assert_eq!(cmd.alias.unwrap(), "t2"); + assert_eq!(cmd.hotkey.unwrap(), "t2"); + + /* ---------------------- Delete command by id ---------------------- */ + db.delete_command_by_id(1).unwrap(); + let cmds = db.get_commands_by_ext_id(ext.ext_id).unwrap(); + assert_eq!(cmds.len(), 1); + + /* ---------------------- Update command by id ---------------------- */ + db.update_command_by_id( + cmds[0].cmd_id, + "test3", + CmdType::UiWorker, + "{}", + false, + Some("alias"), + Some("Command+U"), + ) + .unwrap(); + let cmd = db.get_command_by_id(cmds[0].cmd_id).unwrap().unwrap(); + assert_eq!(cmd.name, "test3"); + assert_eq!(cmd.type_, models::CmdType::UiWorker); + assert_eq!(cmd.data, "{}"); + assert_eq!(cmd.enabled, false); + assert_eq!(cmd.alias.unwrap(), "alias"); + assert_eq!(cmd.hotkey.unwrap(), "Command+U"); + + fs::remove_file(&db_path).unwrap(); + } +} diff --git a/packages/db/src/main.rs b/packages/db/src/main.rs new file mode 100644 index 0000000..9694457 --- /dev/null +++ b/packages/db/src/main.rs @@ -0,0 +1,13 @@ +use db::JarvisDB; +use rusqlite::{params, Connection, Result}; + +fn main() -> Result<()> { + let db = JarvisDB::new("jarvis.db", None)?; + db.init()?; + db.create_extension("test", "0.1.0", true, None, None)?; + let plugins = db.get_all_extensions()?; + for plugin in plugins { + println!("{:#?}", plugin); + } + Ok(()) +} diff --git a/packages/db/src/models.rs b/packages/db/src/models.rs new file mode 100644 index 0000000..8e01968 --- /dev/null +++ b/packages/db/src/models.rs @@ -0,0 +1,89 @@ +use rusqlite::types::FromSql; +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Serialize, Deserialize, Clone)] +#[serde(rename_all = "camelCase")] +pub struct Ext { + pub ext_id: i32, + pub identifier: String, + pub path: Option, + pub data: Option, + pub version: String, + pub enabled: bool, + pub installed_at: String, +} + +#[derive(Debug, Serialize, Deserialize, Clone)] +#[serde(rename_all = "camelCase")] +pub struct ExtData { + pub data_id: i32, + pub ext_id: i32, + pub data_type: String, + pub data: Option, + pub search_text: Option, + pub created_at: String, + pub updated_at: String, +} + +#[derive(Debug, Serialize, Deserialize, Clone, PartialEq)] +#[serde(rename_all = "snake_case")] +pub enum CmdType { + Iframe, + UiWorker, + HeadlessWorker, + QuickLink, + Remote, +} + +impl CmdType { + pub fn to_string(&self) -> String { + serde_json::to_string(self) + .map(|s| s.trim_matches('"').to_string()) + .unwrap_or_else(|_| String::from("")) + } +} + +impl FromSql for CmdType { + fn column_result(value: rusqlite::types::ValueRef) -> rusqlite::types::FromSqlResult { + let type_: String = value.as_str()?.to_string(); + match type_.as_str() { + "iframe" => Ok(CmdType::Iframe), + "ui_worker" => Ok(CmdType::UiWorker), + "headless_worker" => Ok(CmdType::HeadlessWorker), + "quick_link" => Ok(CmdType::QuickLink), + "remote" => Ok(CmdType::Remote), + _ => Err(rusqlite::types::FromSqlError::InvalidType), + } + } +} + +#[derive(Debug, Serialize, Deserialize, Clone)] +#[serde(rename_all = "camelCase")] +pub struct Cmd { + pub cmd_id: i32, + pub ext_id: i32, + pub name: String, + #[serde(rename = "type")] + pub type_: CmdType, + pub data: String, + pub alias: Option, + pub hotkey: Option, + pub enabled: bool, +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_cmd_type() { + let iframe = CmdType::Iframe; + let worker = CmdType::UiWorker; + let headless_worker = CmdType::HeadlessWorker; + let quick_link = CmdType::QuickLink; + assert_eq!(iframe.to_string(), "iframe"); + assert_eq!(worker.to_string(), "ui_worker"); + assert_eq!(headless_worker.to_string(), "headless_worker"); + assert_eq!(quick_link.to_string(), "quick_link"); + } +} diff --git a/packages/db/src/schema.rs b/packages/db/src/schema.rs new file mode 100644 index 0000000..229671b --- /dev/null +++ b/packages/db/src/schema.rs @@ -0,0 +1,20 @@ +pub const SCHEMA1: &str = include_str!("../sql/2024-10-23.sql"); +pub struct Migration { + pub version: u16, + pub script: String, + pub description: String, +} +impl Migration { + pub fn new(version: u16, schema: &str, description: &str) -> Self { + Self { + version, + script: schema.to_string(), + description: description.to_string(), + } + } +} + +use std::sync::LazyLock; + +pub static MIGRATIONS: LazyLock> = + LazyLock::new(|| vec![Migration::new(1, SCHEMA1, "Initial Migration")]); diff --git a/packages/extension/.gitignore b/packages/extension/.gitignore new file mode 100644 index 0000000..9b1ee42 --- /dev/null +++ b/packages/extension/.gitignore @@ -0,0 +1,175 @@ +# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs + +logs +_.log +npm-debug.log_ +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Caches + +.cache + +# Diagnostic reports (https://nodejs.org/api/report.html) + +report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json + +# Runtime data + +pids +_.pid +_.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover + +lib-cov + +# Coverage directory used by tools like istanbul + +coverage +*.lcov + +# nyc test coverage + +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) + +.grunt + +# Bower dependency directory (https://bower.io/) + +bower_components + +# node-waf configuration + +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) + +build/Release + +# Dependency directories + +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) + +web_modules/ + +# TypeScript cache + +*.tsbuildinfo + +# Optional npm cache directory + +.npm + +# Optional eslint cache + +.eslintcache + +# Optional stylelint cache + +.stylelintcache + +# Microbundle cache + +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history + +.node_repl_history + +# Output of 'npm pack' + +*.tgz + +# Yarn Integrity file + +.yarn-integrity + +# dotenv environment variable files + +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) + +.parcel-cache + +# Next.js build output + +.next +out + +# Nuxt.js build / generate output + +.nuxt +dist + +# Gatsby files + +# Comment in the public line in if your project uses Gatsby and not Next.js + +# https://nextjs.org/blog/next-9-1#public-directory-support + +# public + +# vuepress build output + +.vuepress/dist + +# vuepress v2.x temp and cache directory + +.temp + +# Docusaurus cache and generated files + +.docusaurus + +# Serverless directories + +.serverless/ + +# FuseBox cache + +.fusebox/ + +# DynamoDB Local files + +.dynamodb/ + +# TernJS port file + +.tern-port + +# Stores VSCode versions used for testing VSCode extensions + +.vscode-test + +# yarn v2 + +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# IntelliJ based IDEs +.idea + +# Finder (MacOS) folder config +.DS_Store diff --git a/packages/extension/package.json b/packages/extension/package.json new file mode 100644 index 0000000..3c19a15 --- /dev/null +++ b/packages/extension/package.json @@ -0,0 +1,25 @@ +{ + "name": "@kksh/extension", + "module": "index.ts", + "type": "module", + "exports": { + ".": "./src/index.ts", + "./db": "./src/db.ts", + "./load": "./src/load.ts", + "./utils": "./src/utils.ts", + "./window": "./src/window.ts", + "./install": "./src/install.ts" + }, + "devDependencies": { + "@types/bun": "latest" + }, + "dependencies": { + "@kksh/api": "workspace:*", + "@kksh/supabase": "workspace:*", + "@std/semver": "npm:@jsr/std__semver@^1.0.3", + "uuid": "^11.0.2" + }, + "peerDependencies": { + "typescript": "^5.0.0" + } +} diff --git a/packages/extension/src/db.ts b/packages/extension/src/db.ts new file mode 100644 index 0000000..a6246ba --- /dev/null +++ b/packages/extension/src/db.ts @@ -0,0 +1,14 @@ +import { db } from "@kksh/api/commands" +import { ExtPackageJson, ExtPackageJsonExtra } from "@kksh/api/models" + +export async function upsertExtension(extPkgJson: ExtPackageJson, extFullPath: string) { + const extInDb = await db.getUniqueExtensionByIdentifier(extPkgJson.kunkun.identifier) + if (!extInDb) { + // create this extension in database + await db.createExtension({ + identifier: extPkgJson.kunkun.identifier, + version: extPkgJson.version, + path: extFullPath + }) + } +} diff --git a/packages/extension/src/index.ts b/packages/extension/src/index.ts new file mode 100644 index 0000000..a279545 --- /dev/null +++ b/packages/extension/src/index.ts @@ -0,0 +1,5 @@ +export * from "./db" +export * from "./load" +export * from "./utils" +export * from "./window" +export * from "./install" diff --git a/packages/extension/src/install.ts b/packages/extension/src/install.ts new file mode 100644 index 0000000..0704f26 --- /dev/null +++ b/packages/extension/src/install.ts @@ -0,0 +1,155 @@ +/** + * This file contains helper functions for operations related to installing extensions + * including install, uninstall, upgrade, check app-extension compatibility, etc. + */ +import { isCompatible } from "@kksh/api" +import { db, decompressTarball } from "@kksh/api/commands" +import type { ExtPackageJsonExtra } from "@kksh/api/models" +import { SBExt } from "@kksh/supabase" +import { greaterThan, parse as parseSemver } from "@std/semver" +import * as path from "@tauri-apps/api/path" +import * as dialog from "@tauri-apps/plugin-dialog" +import * as fs from "@tauri-apps/plugin-fs" +import { download } from "@tauri-apps/plugin-upload" +import { v4 as uuidv4 } from "uuid" +import { z, ZodError } from "zod" +import { loadExtensionManifestFromDisk } from "./load" + +/** + * + * @param tarballPath path to .tar.gz file + */ +export async function installTarball(tarballPath: string, extsDir: string): Promise { + const tempDirPath = await path.tempDir() + if (!extsDir) { + return Promise.reject("Extension Folder Not Set") + } + // decompress tarball to tempDir + const decompressDest = await decompressTarball( + tarballPath, + await path.join(tempDirPath, uuidv4()), + { + overwrite: true + } + ) + return loadExtensionManifestFromDisk(await path.join(decompressDest, "package.json")) + .then(async (manifest) => { + // The extension folder name will be the identifier + const extInstallPath = await path.join(extsDir, manifest.kunkun.identifier) + if (await fs.exists(extInstallPath)) { + const overwrite = await dialog.ask( + `Extension ${manifest.kunkun.identifier} already exists, do you want to overwrite it?` + ) + if (!overwrite) { + return Promise.reject("Extension Already Exists") + } + await fs.remove(extInstallPath, { recursive: true }) + } + await fs.rename(decompressDest, extInstallPath) + await db.createExtension({ + identifier: manifest.kunkun.identifier, + version: manifest.version, + enabled: true, + path: extInstallPath, + data: undefined + }) + + console.log("installTarball in DB success", manifest) + return extInstallPath + }) + .catch((err) => { + if (err instanceof ZodError) { + console.error(err) + throw new Error("Invalid Manifest or Extension") + } + console.log() + + throw new Error(err) + }) +} + +/** + * Install extension tarball from a URL + * @param tarballUrl URL to the tarball + * @param extsDir Target directory to install the tarball + * @returns + */ +export async function installTarballUrl(tarballUrl: string, extsDir: string): Promise { + const filename = await path.basename(tarballUrl) + if (filename) { + const tempDirPath = await path.tempDir() + let tarballPath = await path.join(tempDirPath, filename) + await download(tarballUrl, tarballPath) + const extInstallPath = await installTarball(tarballPath, extsDir) + await fs.remove(tarballPath) + return extInstallPath + } else { + return Promise.reject("Invalid Tarball URL. Cannot parse filename") + } +} + +export async function installDevExtensionDir(extPath: string): Promise { + const manifestPath = await path.join(extPath, "package.json") + if (!(await fs.exists(manifestPath))) { + return Promise.reject( + `Invalid Extension Folder. Manifest package.json doesn't exist at ${manifestPath}` + ) + } + return loadExtensionManifestFromDisk(manifestPath) + .then(async (manifest) => { + const exts = await db.getAllExtensionsByIdentifier(manifest.kunkun.identifier) + const extExists = exts.find((ext) => ext.path === extPath) + if (extExists) { + return Promise.reject(`Extension Already Exists at ${extExists.path}. It will be skipped.`) + } + // manifest.extPath + return db + .createExtension({ + identifier: manifest.kunkun.identifier, + version: manifest.version, + enabled: true, + path: extPath, + data: undefined + }) + .then(() => { + return manifest + }) + .catch((err) => { + return Promise.reject(err) + }) + }) + .catch((err) => { + return Promise.reject(err) + }) +} + +export async function installThroughNpmAPI(url: string, extsDir: string) { + return fetch(url) + .then((res) => res.json()) + .then((json) => { + const tarball = z.string().safeParse(json?.dist?.tarball) + if (tarball.success) { + return installTarballUrl(tarball.data, extsDir) + } else { + return Promise.reject("Tarball Not Found") + } + }) +} + +export async function installFromNpmPackageName(name: string, extsDir: string) { + return installThroughNpmAPI(`https://registry.npmjs.org/${name}/latest`, extsDir) +} + +export async function uninstallExtensionByPath(extPath: string) { + if (!(await fs.exists(extPath))) throw new Error(`Extension ${extPath} not found`) + return fs.remove(extPath, { recursive: true }).then(() => db.deleteExtensionByPath(extPath)) +} + +export function isUpgradable(dbExt: SBExt, installedExtVersion: string) { + const upgradable = + greaterThan(parseSemver(dbExt.version), parseSemver(installedExtVersion)) && dbExt.api_version + ? isCompatible(dbExt.api_version) + : false + + return upgradable +} diff --git a/packages/extension/src/load.ts b/packages/extension/src/load.ts new file mode 100644 index 0000000..4371d4f --- /dev/null +++ b/packages/extension/src/load.ts @@ -0,0 +1,73 @@ +import { db } from "@kksh/api/commands" +import { ExtPackageJson, ExtPackageJsonExtra } from "@kksh/api/models" +import { basename, dirname, join } from "@tauri-apps/api/path" +import { readDir, readTextFile } from "@tauri-apps/plugin-fs" +import { debug, error } from "@tauri-apps/plugin-log" +import { flatten, safeParse } from "valibot" +import { upsertExtension } from "./db" + +/** + * + * @param manifestPath absolute path to package.json + * @returns + */ +export function loadExtensionManifestFromDisk(manifestPath: string): Promise { + debug(`loadExtensionManifestFromDisk: ${manifestPath}`) + return readTextFile(manifestPath).then(async (content) => { + const parse = safeParse(ExtPackageJson, JSON.parse(content)) + if (parse.issues) { + error(`Fail to load extension from ${manifestPath}. See console for parse error.`) + console.error(parse.issues) + console.error(JSON.stringify(flatten(parse.issues), null, 2)) + throw new Error(`Invalid manifest: ${manifestPath}`) + } else { + // debug(`Loaded extension ${parse.output.kunkun.identifier} from ${manifestPath}`) + const extPath = await dirname(manifestPath) + const extFolderName = await basename(extPath) + return Object.assign(parse.output, { + extPath, + extFolderName + }) + } + }) +} +export function loadAllExtensionsFromDisk( + extensionsFolder: string +): Promise { + return readDir(extensionsFolder).then((dirEntries) => { + return Promise.all( + dirEntries.map(async (dirEntry) => { + const extFullPath = await join(extensionsFolder, dirEntry.name) + const manifestPath = await join(extFullPath, "package.json") + try { + const extPkgJson = await loadExtensionManifestFromDisk(manifestPath) + await upsertExtension(extPkgJson, extFullPath) + return Object.assign(extPkgJson, { + extPath: extFullPath, + extFolderName: dirEntry.name + }) + } catch (error) { + return null + } + }) + ).then((results) => results.filter((r): r is ExtPackageJsonExtra => r !== null)) + }) +} + +export async function loadAllExtensionsFromDb(): Promise { + const allDbExts = await (await db.getAllExtensions()).filter((ext) => ext.path) + const results: ExtPackageJsonExtra[] = [] + for (const ext of allDbExts) { + if (!ext.path) continue + try { + const extPkgJson = await loadExtensionManifestFromDisk(await join(ext.path, "package.json")) + results.push(extPkgJson) + } catch (err) { + console.error(err) + error(`Failed to load extension ${ext.path} from database.`) + // delete this extension from database + await db.deleteExtensionByPath(ext.path) + } + } + return results +} diff --git a/packages/extension/src/utils.ts b/packages/extension/src/utils.ts new file mode 100644 index 0000000..6dc560b --- /dev/null +++ b/packages/extension/src/utils.ts @@ -0,0 +1,9 @@ +/** + * Check if the extension path is in dev mode + * @param extPath - The main extension path + * @param candidateExtPath - The candidate extension path + * @returns True if the extension path is in dev mode, false otherwise + */ +export function isExtPathInDev(extPath: string, candidateExtPath: string) { + return !candidateExtPath.startsWith(extPath) +} diff --git a/packages/extension/src/window.ts b/packages/extension/src/window.ts new file mode 100644 index 0000000..26790f0 --- /dev/null +++ b/packages/extension/src/window.ts @@ -0,0 +1,39 @@ +import type { WindowConfig } from "@kksh/api/models" +import { WebviewWindow } from "@tauri-apps/api/webviewWindow" + +export function launchNewExtWindow(windowLabel: string, url: string, windowConfig?: WindowConfig) { + return new WebviewWindow(windowLabel, { + center: windowConfig?.center ?? undefined, + x: windowConfig?.x ?? undefined, + y: windowConfig?.y ?? undefined, + width: windowConfig?.width ?? undefined, + height: windowConfig?.height ?? undefined, + minWidth: windowConfig?.minWidth ?? undefined, + minHeight: windowConfig?.minHeight ?? undefined, + maxWidth: windowConfig?.maxWidth ?? undefined, + maxHeight: windowConfig?.maxHeight ?? undefined, + resizable: windowConfig?.resizable ?? undefined, + title: windowConfig?.title ?? undefined, + fullscreen: windowConfig?.fullscreen ?? undefined, + focus: windowConfig?.focus ?? undefined, + transparent: windowConfig?.transparent ?? undefined, + maximized: windowConfig?.maximized ?? undefined, + visible: windowConfig?.visible ?? false, // default to false to avoid flickering + decorations: windowConfig?.decorations ?? undefined, + alwaysOnTop: windowConfig?.alwaysOnTop ?? undefined, + alwaysOnBottom: windowConfig?.alwaysOnBottom ?? undefined, + contentProtected: windowConfig?.contentProtected ?? undefined, + skipTaskbar: windowConfig?.skipTaskbar ?? undefined, + shadow: windowConfig?.shadow ?? undefined, + // theme: windowConfig?.theme ?? undefined, + titleBarStyle: windowConfig?.titleBarStyle ?? undefined, + hiddenTitle: windowConfig?.hiddenTitle ?? undefined, + tabbingIdentifier: windowConfig?.tabbingIdentifier ?? undefined, + maximizable: windowConfig?.maximizable ?? undefined, + minimizable: windowConfig?.minimizable ?? undefined, + closable: windowConfig?.closable ?? undefined, + parent: windowConfig?.parent ?? undefined, + visibleOnAllWorkspaces: windowConfig?.visibleOnAllWorkspaces ?? undefined, + url + }) +} diff --git a/packages/extension/tsconfig.json b/packages/extension/tsconfig.json new file mode 100644 index 0000000..238655f --- /dev/null +++ b/packages/extension/tsconfig.json @@ -0,0 +1,27 @@ +{ + "compilerOptions": { + // Enable latest features + "lib": ["ESNext", "DOM"], + "target": "ESNext", + "module": "ESNext", + "moduleDetection": "force", + "jsx": "react-jsx", + "allowJs": true, + + // Bundler mode + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "noEmit": true, + + // Best practices + "strict": true, + "skipLibCheck": true, + "noFallthroughCasesInSwitch": true, + + // Some stricter flags (disabled by default) + "noUnusedLocals": false, + "noUnusedParameters": false, + "noPropertyAccessFromIndexSignature": false + } +} diff --git a/packages/extensions/.gitkeep b/packages/extensions/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/packages/mac-security-rs/Cargo.toml b/packages/mac-security-rs/Cargo.toml new file mode 100644 index 0000000..5b713f9 --- /dev/null +++ b/packages/mac-security-rs/Cargo.toml @@ -0,0 +1,14 @@ +[package] +name = "mac-security-rs" +version = "0.1.0" +edition = "2021" +build = "build.rs" + +[target.'cfg(target_os = "macos")'.dependencies] +serde = { workspace = true, features = ["derive"] } +core-foundation = "0.10.0" +localauthentication-rs = "0.1.0" +objc = "0.2.7" + +[package.metadata] +target_os = "macos" diff --git a/packages/mac-security-rs/build.rs b/packages/mac-security-rs/build.rs new file mode 100644 index 0000000..df23ff8 --- /dev/null +++ b/packages/mac-security-rs/build.rs @@ -0,0 +1,7 @@ +fn main() { + #[cfg(not(target_os = "macos"))] + { + println!("cargo:warning=This crate is only intended for macOS systems."); + std::process::exit(0); + } +} diff --git a/packages/mac-security-rs/src/lib.rs b/packages/mac-security-rs/src/lib.rs new file mode 100644 index 0000000..cacdec5 --- /dev/null +++ b/packages/mac-security-rs/src/lib.rs @@ -0,0 +1,57 @@ +#![cfg(target_os = "macos")] + +use localauthentication_rs::{LAPolicy, LocalAuthentication}; +use objc::runtime::{Object, BOOL, NO, YES}; +use objc::{msg_send, sel, sel_impl}; +use serde::{Deserialize, Serialize}; +use std::ptr; + +#[link(name = "CoreGraphics", kind = "framework")] +extern "C" { + fn CGRequestScreenCaptureAccess() -> BOOL; + fn CGPreflightScreenCaptureAccess() -> BOOL; +} + +pub fn request_screen_capture_access() -> bool { + unsafe { + let result: BOOL = CGRequestScreenCaptureAccess(); + result == YES + } +} + +/// Check if we already have screen capture access +/// Returns true if we have access, false otherwise +pub fn preflight_screen_capture_access() -> bool { + unsafe { + let result: BOOL = CGPreflightScreenCaptureAccess(); + result == YES + } +} + +#[derive(Clone, Copy, Debug, PartialEq, Eq, Serialize, Deserialize)] +pub enum AuthPolicy { + Any, + Biometrics, + Watch, + BiometricsOrWatch, +} + +pub fn verify_auth(policy: AuthPolicy) -> bool { + let local_authentication = LocalAuthentication::new(); + + // Try to authenticate the user + let authenticated = local_authentication.evaluate_policy( + // LAPolicy::DeviceOwnerAuthenticationWithBiometrics, + match policy { + AuthPolicy::Any => LAPolicy::DeviceOwnerAuthentication, + AuthPolicy::Biometrics => LAPolicy::DeviceOwnerAuthenticationWithBiometrics, + AuthPolicy::Watch => LAPolicy::DeviceOwnerAuthenticationWithWatch, + AuthPolicy::BiometricsOrWatch => { + LAPolicy::DeviceOwnerAuthenticationWithBiometricsOrWatch + } + }, + "authenticate your user", + ); + + authenticated +} diff --git a/packages/mac-security-rs/src/main.rs b/packages/mac-security-rs/src/main.rs new file mode 100644 index 0000000..00a2083 --- /dev/null +++ b/packages/mac-security-rs/src/main.rs @@ -0,0 +1,13 @@ +#![cfg(target_os = "macos")] +use mac_security_rs::{ + preflight_screen_capture_access, request_screen_capture_access, verify_auth, AuthPolicy, +}; + +fn main() { + #[cfg(target_os = "macos")] + // println!("{}", request_screen_capture_access()); + println!("{}", verify_auth(AuthPolicy::Biometrics)); +} + +#[cfg(not(target_os = "macos"))] +fn main() {} diff --git a/packages/schema/.gitignore b/packages/schema/.gitignore new file mode 100644 index 0000000..1f3a91a --- /dev/null +++ b/packages/schema/.gitignore @@ -0,0 +1,176 @@ +# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs + +logs +_.log +npm-debug.log_ +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Caches + +.cache + +# Diagnostic reports (https://nodejs.org/api/report.html) + +report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json + +# Runtime data + +pids +_.pid +_.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover + +lib-cov + +# Coverage directory used by tools like istanbul + +coverage +*.lcov + +# nyc test coverage + +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) + +.grunt + +# Bower dependency directory (https://bower.io/) + +bower_components + +# node-waf configuration + +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) + +build/Release + +# Dependency directories + +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) + +web_modules/ + +# TypeScript cache + +*.tsbuildinfo + +# Optional npm cache directory + +.npm + +# Optional eslint cache + +.eslintcache + +# Optional stylelint cache + +.stylelintcache + +# Microbundle cache + +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history + +.node_repl_history + +# Output of 'npm pack' + +*.tgz + +# Yarn Integrity file + +.yarn-integrity + +# dotenv environment variable files + +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) + +.parcel-cache + +# Next.js build output + +.next +out + +# Nuxt.js build / generate output + +.nuxt +dist + +# Gatsby files + +# Comment in the public line in if your project uses Gatsby and not Next.js + +# https://nextjs.org/blog/next-9-1#public-directory-support + +# public + +# vuepress build output + +.vuepress/dist + +# vuepress v2.x temp and cache directory + +.temp + +# Docusaurus cache and generated files + +.docusaurus + +# Serverless directories + +.serverless/ + +# FuseBox cache + +.fusebox/ + +# DynamoDB Local files + +.dynamodb/ + +# TernJS port file + +.tern-port + +# Stores VSCode versions used for testing VSCode extensions + +.vscode-test + +# yarn v2 + +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# IntelliJ based IDEs +.idea + +# Finder (MacOS) folder config +.DS_Store +manifest-json-schema.json diff --git a/packages/schema/README.md b/packages/schema/README.md new file mode 100644 index 0000000..17bab78 --- /dev/null +++ b/packages/schema/README.md @@ -0,0 +1,14 @@ +```bash +npx supabase gen types typescript --project-id "$PROJECT_REF" --schema public > supabase/types/supabase.ts + +``` + +Add the following to `.env`: + +```bash +SUPABASE_URL= +SUPABASE_SERVICE_ROLE_KEY= +S3_ENDPOINT= +S3_ACCESS_KEY_ID= +S3_SECRET_ACCESS_KEY= +``` diff --git a/packages/schema/__tests__/schema.test.ts b/packages/schema/__tests__/schema.test.ts new file mode 100644 index 0000000..685b8da --- /dev/null +++ b/packages/schema/__tests__/schema.test.ts @@ -0,0 +1,7 @@ +import { ExtPackageJson } from "@kksh/api/models" +import { describe, expect, it } from "bun:test" +import { getJsonSchema } from "../src" + +it("Generate Schema", () => { + expect(getJsonSchema(ExtPackageJson)).toBeDefined() +}) diff --git a/packages/schema/package.json b/packages/schema/package.json new file mode 100644 index 0000000..e3070c8 --- /dev/null +++ b/packages/schema/package.json @@ -0,0 +1,31 @@ +{ + "name": "@kksh/schema", + "type": "module", + "exports": { + ".": { + "import": "./index.ts" + } + }, + "scripts": { + "build": "bun scripts/print-schema.ts > manifest-json-schema.json", + "test": "bun test --coverage", + "upload-schema-to-supabase": "bun scripts/upload-schema-to-supabase.ts", + "upload-schema-to-s3": "bun scripts/upload-schema-to-s3.ts", + "print-schema": "bun scripts/print-schema.ts" + }, + "devDependencies": { + "@gcornut/valibot-json-schema": "^0.42.0", + "@types/bun": "latest", + "supabase": ">=1.8.1" + }, + "peerDependencies": { + "@kksh/supabase": "workspace:*", + "typescript": "^5.0.0" + }, + "dependencies": { + "@aws-sdk/client-s3": "^3.583.0", + "@kksh/api": "workspace:*", + "@supabase/supabase-js": "^2.43.4", + "valibot": "^0.40.0" + } +} diff --git a/packages/schema/scripts/print-schema.ts b/packages/schema/scripts/print-schema.ts new file mode 100644 index 0000000..a556c7e --- /dev/null +++ b/packages/schema/scripts/print-schema.ts @@ -0,0 +1,4 @@ +import { ExtPackageJson } from "@kksh/api/models" +import { getJsonSchema } from "../src" + +console.log(getJsonSchema(ExtPackageJson)) diff --git a/packages/schema/scripts/upload-schema-to-s3.ts b/packages/schema/scripts/upload-schema-to-s3.ts new file mode 100644 index 0000000..ad8022c --- /dev/null +++ b/packages/schema/scripts/upload-schema-to-s3.ts @@ -0,0 +1,65 @@ +import { + GetObjectCommand, + ListBucketsCommand, + PutObjectCommand, + S3Client +} from "@aws-sdk/client-s3" +import { toJSONSchema } from "@gcornut/valibot-json-schema" +import { ExtPackageJson } from "@kksh/api/models" +import { parse, string } from "valibot" +import { getJsonSchema } from "../src" + +const s3Client = new S3Client({ + endpoint: parse(string(), process.env.S3_ENDPOINT), + region: "auto", + credentials: { + accessKeyId: parse(string(), process.env.S3_ACCESS_KEY_ID), + secretAccessKey: parse(string(), process.env.S3_SECRET_ACCESS_KEY) + } +}) +/* -------------------------------------------------------------------------- */ +/* Get Schema */ +/* -------------------------------------------------------------------------- */ +// const { Body } = await s3Client.send( +// new GetObjectCommand({ +// Bucket: "jarvis-extensions", +// Key: "schema.json", +// }), +// ); +// const data = await Body?.transformToByteArray(); +// if (data) { +// fs.writeFileSync("schema.json", data); +// } + +/* -------------------------------------------------------------------------- */ +/* Upload Schema to S3 */ +/* -------------------------------------------------------------------------- */ + +const schemaStr = getJsonSchema(ExtPackageJson) + +await s3Client.send( + new PutObjectCommand({ + Bucket: "jarvis-extensions", + Key: "nightly.schema.json", + Body: schemaStr, + ContentType: "application/json" + }) +) + +s3Client + .send( + new PutObjectCommand({ + Bucket: "jarvis-extensions", + Key: "schema.json", + Body: schemaStr, + ContentType: "application/json" + }) + ) + .then(() => { + console.log("Schema uploaded to S3") + }) + .catch((err) => { + console.error("Failed to upload schema.json") + console.error(err) + process.exit(1) + }) diff --git a/packages/schema/scripts/upload-schema-to-supabase.ts b/packages/schema/scripts/upload-schema-to-supabase.ts new file mode 100644 index 0000000..f4bf48a --- /dev/null +++ b/packages/schema/scripts/upload-schema-to-supabase.ts @@ -0,0 +1,27 @@ +import { ExtPackageJson } from "@kksh/api/models" +import { type Database } from "@kksh/supabase" +import { createClient } from "@supabase/supabase-js" +import { parse, string } from "valibot" +import { getJsonSchema } from "../src" + +const supabase = createClient( + parse(string(), process.env.SUPABASE_URL), + parse(string(), process.env.SUPABASE_SERVICE_ROLE_KEY) +) + +const schemaStr = getJsonSchema(ExtPackageJson) + +const { data, error } = await supabase.storage.from("extensions").upload("schema.json", schemaStr, { + cacheControl: "3600", + upsert: true // overwrite existing file with same name +}) +await supabase.storage.from("extensions").upload("nightly.schema.json", schemaStr, { + cacheControl: "3600", + upsert: true // overwrite existing file with same name +}) +console.log("data", data) +if (error) { + console.error("Failed to upload schema.json") + console.error(error) + process.exit(1) +} diff --git a/packages/schema/src/index.ts b/packages/schema/src/index.ts new file mode 100644 index 0000000..99a3fb2 --- /dev/null +++ b/packages/schema/src/index.ts @@ -0,0 +1,5 @@ +import { toJSONSchema } from "@gcornut/valibot-json-schema" + +export function getJsonSchema(schema: any) { + return JSON.stringify(toJSONSchema({ schema }), null, 2) +} diff --git a/packages/schema/tsconfig.json b/packages/schema/tsconfig.json new file mode 100644 index 0000000..ffc08ab --- /dev/null +++ b/packages/schema/tsconfig.json @@ -0,0 +1,27 @@ +{ + "compilerOptions": { + // Enable latest features + "lib": ["ESNext", "DOM"], + "target": "ESNext", + "module": "ESNext", + "moduleDetection": "force", + "jsx": "react-jsx", + "allowJs": true, + + // Bundler mode + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "noEmit": true, + + // Best practices + "strict": true, + "skipLibCheck": true, + "noFallthroughCasesInSwitch": true, + + // Some stricter flags (disabled by default) + "noUnusedLocals": false, + "noUnusedParameters": false, + "noPropertyAccessFromIndexSignature": false + } +} diff --git a/packages/supabase/.gitignore b/packages/supabase/.gitignore new file mode 100644 index 0000000..9b1ee42 --- /dev/null +++ b/packages/supabase/.gitignore @@ -0,0 +1,175 @@ +# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs + +logs +_.log +npm-debug.log_ +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Caches + +.cache + +# Diagnostic reports (https://nodejs.org/api/report.html) + +report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json + +# Runtime data + +pids +_.pid +_.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover + +lib-cov + +# Coverage directory used by tools like istanbul + +coverage +*.lcov + +# nyc test coverage + +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) + +.grunt + +# Bower dependency directory (https://bower.io/) + +bower_components + +# node-waf configuration + +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) + +build/Release + +# Dependency directories + +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) + +web_modules/ + +# TypeScript cache + +*.tsbuildinfo + +# Optional npm cache directory + +.npm + +# Optional eslint cache + +.eslintcache + +# Optional stylelint cache + +.stylelintcache + +# Microbundle cache + +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history + +.node_repl_history + +# Output of 'npm pack' + +*.tgz + +# Yarn Integrity file + +.yarn-integrity + +# dotenv environment variable files + +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) + +.parcel-cache + +# Next.js build output + +.next +out + +# Nuxt.js build / generate output + +.nuxt +dist + +# Gatsby files + +# Comment in the public line in if your project uses Gatsby and not Next.js + +# https://nextjs.org/blog/next-9-1#public-directory-support + +# public + +# vuepress build output + +.vuepress/dist + +# vuepress v2.x temp and cache directory + +.temp + +# Docusaurus cache and generated files + +.docusaurus + +# Serverless directories + +.serverless/ + +# FuseBox cache + +.fusebox/ + +# DynamoDB Local files + +.dynamodb/ + +# TernJS port file + +.tern-port + +# Stores VSCode versions used for testing VSCode extensions + +.vscode-test + +# yarn v2 + +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# IntelliJ based IDEs +.idea + +# Finder (MacOS) folder config +.DS_Store diff --git a/packages/supabase/README.md b/packages/supabase/README.md new file mode 100644 index 0000000..f714f66 --- /dev/null +++ b/packages/supabase/README.md @@ -0,0 +1,3 @@ +```bash +npx supabase gen types --lang=typescript --project-id $PROJECT_REF --schema public > src/types/database.types.ts +``` diff --git a/packages/supabase/package.json b/packages/supabase/package.json new file mode 100644 index 0000000..4248fdd --- /dev/null +++ b/packages/supabase/package.json @@ -0,0 +1,19 @@ +{ + "name": "@kksh/supabase", + "type": "module", + "scripts": { + "prepare": "bun setup.ts" + }, + "exports": { + ".": "./src/index.ts" + }, + "dependencies": { + "@kksh/api": "workspace:*" + }, + "devDependencies": { + "@types/bun": "latest" + }, + "peerDependencies": { + "typescript": "^5.0.0" + } +} diff --git a/packages/supabase/setup.ts b/packages/supabase/setup.ts new file mode 100644 index 0000000..e69de29 diff --git a/packages/supabase/src/api.ts b/packages/supabase/src/api.ts new file mode 100644 index 0000000..a78cbde --- /dev/null +++ b/packages/supabase/src/api.ts @@ -0,0 +1,76 @@ +import { SBExt } from "@kksh/api/supabase" +import type { Database, Tables } from "@kksh/api/supabase/types" +import type { PostgrestSingleResponse, SupabaseClient } from "@supabase/supabase-js" +import * as v from "valibot" + +export class SupabaseAPI { + constructor(private supabase: SupabaseClient) {} + async getExtList(): Promise { + const res = await this.supabase + .from("extensions") + .select( + "identifier, version, api_version, name, downloads, short_description, long_description, icon, created_at" + ) + .order("downloads", { ascending: false }) + .select() + const dbExts: Tables<"extensions">[] = res.data ?? [] + return dbExts + .map((x) => { + const parsedNode = v.safeParse(SBExt, x) + if (!parsedNode.success) { + console.error(`Fail to parse extension`, x) + console.warn(parsedNode.issues) + console.error(v.flatten(parsedNode.issues)) + } + return parsedNode.success ? v.parse(SBExt, parsedNode.output) : null + }) + .filter((x) => x !== null) + } + + async getLatestExtPublish( + identifier: string + ): Promise>> { + return this.supabase + .from("ext_publish") + .select( + "created_at, name, version, manifest, shasum, size, tarball_path, cmd_count, identifier, downloads, demo_images, api_version" + ) + .order("created_at", { ascending: false }) + .eq("identifier", identifier) + .select() + .limit(1) + .single() + } + + async incrementDownloads({ + identifier, + version + }: { + identifier: string + version: string + }): Promise<{ downloads: number }> { + return this.supabase.functions + .invoke("increment-downloads", { + body: { identifier, version } + }) + .then(({ data, error }) => { + if (error) { + throw error + } + const parsed = v.safeParse( + v.object({ + downloads: v.number() + }), + data + ) + if (!parsed.success) { + throw new Error("Fail to parse increment downloads response") + } + return parsed.output + }) + } + + translateExtensionFilePathToUrl(tarballPath: string): string { + return this.supabase.storage.from("extensions").getPublicUrl(tarballPath).data.publicUrl + } +} diff --git a/packages/supabase/src/index.ts b/packages/supabase/src/index.ts new file mode 100644 index 0000000..9285d65 --- /dev/null +++ b/packages/supabase/src/index.ts @@ -0,0 +1,10 @@ +import type { Database } from "@kksh/api/supabase/types" +import { createClient } from "@supabase/supabase-js" + +export function createSB(supabaseUrl: string, supabaseAnonKey: string) { + return createClient(supabaseUrl, supabaseAnonKey) +} +export { SupabaseAPI } from "./api" + +export type { Database, Tables } from "@kksh/api/supabase/types" +export { SBExt } from "@kksh/api/supabase" diff --git a/packages/supabase/tsconfig.json b/packages/supabase/tsconfig.json new file mode 100644 index 0000000..238655f --- /dev/null +++ b/packages/supabase/tsconfig.json @@ -0,0 +1,27 @@ +{ + "compilerOptions": { + // Enable latest features + "lib": ["ESNext", "DOM"], + "target": "ESNext", + "module": "ESNext", + "moduleDetection": "force", + "jsx": "react-jsx", + "allowJs": true, + + // Bundler mode + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "noEmit": true, + + // Best practices + "strict": true, + "skipLibCheck": true, + "noFallthroughCasesInSwitch": true, + + // Some stricter flags (disabled by default) + "noUnusedLocals": false, + "noUnusedParameters": false, + "noPropertyAccessFromIndexSignature": false + } +} diff --git a/packages/tauri-plugins/jarvis/Cargo.toml b/packages/tauri-plugins/jarvis/Cargo.toml new file mode 100644 index 0000000..20aa4bb --- /dev/null +++ b/packages/tauri-plugins/jarvis/Cargo.toml @@ -0,0 +1,54 @@ +[package] +name = "tauri-plugin-jarvis" +version = "0.0.0" +authors = ["Huakun"] +description = "" +edition = "2021" +rust-version = "1.70" +exclude = ["/examples", "/webview-dist", "/webview-src", "/node_modules"] +links = "tauri-plugin-jarvis" + +[dependencies] +tauri = { version = "2.0.1" } +serde = { workspace = true } +serde_json = { workspace = true } +thiserror = "1.0" +applications = { workspace = true } +anyhow = { workspace = true } +tar = "0.4.40" +flate2 = "1.0.30" +window-vibrancy = "0.5.0" +tauri-plugin-store = "2.0.1" +axum = { version = "0.6.20" } +axum-extra = { version = "0.8.0" } +axum-server = { version = "0.5", features = ["tls-rustls"] } +tower = { version = "0.4", features = ["util"] } +tower-http = { version = "0.4.0", features = ["fs", "trace", "cors"] } +tonic = "0.11" +tonic-reflection = "0.11.0" +uuid = { version = "1.8.0", features = ["v4"] } +prost = "0.12" +tokio = { workspace = true } +mdns-sd = { workspace = true } +tauri-plugin-network = { workspace = true } +db = { workspace = true } +tauri-plugin-clipboard = { workspace = true } +log = { workspace = true } +strum = { workspace = true } +chrono = { workspace = true } +strum_macros = { workspace = true } +mac-security-rs = { workspace = true } +zip = "1.1.4" +rust_search = "2.1.0" +plist = "1.7.0" + +[target.'cfg(target_os = "macos")'.dependencies] +tauri-icns = "0.1.0" + +[target.'cfg(target_os = "windows")'.dependencies] +tauri-winres = "0.1.1" +ico = "0.3.0" + +[build-dependencies] +tauri-plugin = { version = "2.0.1", features = ["build"] } +tonic-build = "0.11" diff --git a/packages/tauri-plugins/jarvis/build.rs b/packages/tauri-plugins/jarvis/build.rs new file mode 100644 index 0000000..c3f87ca --- /dev/null +++ b/packages/tauri-plugins/jarvis/build.rs @@ -0,0 +1,126 @@ +const COMMANDS: &[&str] = &[ + "open_devtools", + "close_devtools", + "is_devtools_open", + "toggle_devtools", + "app_is_dev", + "open_trash", + "empty_trash", + "shutdown", + "reboot", + "sleep", + "toggle_system_appearance", + "show_desktop", + "quit_all_apps", + "sleep_displays", + "set_volume", + "turn_volume_up", + "turn_volume_down", + "toggle_stage_manager", + "toggle_bluetooth", + "toggle_hidden_files", + "eject_all_disks", + "logout_user", + "toggle_mute", + "mute", + "unmute", + "hide_all_apps_except_frontmost", + "get_selected_files_in_file_explorer", + "run_apple_script", + "run_powershell", + "get_applications", + "refresh_applications_list", + "refresh_applications_list_in_bg", + // "load_manifest", + // "load_all_extensions", + "path_exists", + "start_server", + "stop_server", + "restart_server", + "set_dev_extension_folder", + "set_extension_folder", + "get_extension_folder", + "get_dev_extension_folder", + "server_is_running", + /* -------------------------------------------------------------------------- */ + /* FS */ + /* -------------------------------------------------------------------------- */ + "decompress_tarball", + "compress_tarball", + "unzip", + /* -------------------------------------------------------------------------- */ + /* File Search */ + /* -------------------------------------------------------------------------- */ + "file_search", + /* -------------------------------------------------------------------------- */ + /* Path */ + /* -------------------------------------------------------------------------- */ + "get_default_extensions_dir", + "get_default_extensions_storage_dir", + "is_window_label_registered", + "register_extension_window", + "register_extension_spawned_process", + "unregister_extension_spawned_process", + "unregister_extension_window", + "get_ext_label_map", + // "ext_store_wrapper_set", + // "ext_store_wrapper_get", + // "ext_store_wrapper_has", + // "ext_store_wrapper_delete", + // "ext_store_wrapper_clear", + // "ext_store_wrapper_reset", + // "ext_store_wrapper_keys", + // "ext_store_wrapper_values", + // "ext_store_wrapper_entries", + // "ext_store_wrapper_length", + // "ext_store_wrapper_load", + // "ext_store_wrapper_save", + "get_server_port", + /* ----------------------------- sqlite database ---------------------------- */ + "create_extension", + "get_all_extensions", + "get_unique_extension_by_identifier", + "get_unique_extension_by_path", + "get_all_extensions_by_identifier", + "delete_extension_by_path", + "delete_extension_by_ext_id", + "create_command", + "get_command_by_id", + "get_commands_by_ext_id", + "delete_command_by_id", + "update_command_by_id", + "create_extension_data", + "get_extension_data_by_id", + "search_extension_data", + "delete_extension_data_by_id", + "update_extension_data_by_id", + /* -------------------------------- Clipboard ------------------------------- */ + "add_to_history", + "get_history", + /* -------------------------------------------------------------------------- */ + /* Utils */ + /* -------------------------------------------------------------------------- */ + "plist_to_json", + /* -------------------------------------------------------------------------- */ + /* Security */ + /* -------------------------------------------------------------------------- */ + "verify_auth", + "request_screen_capture_access", + "check_screen_capture_access", + /* -------------------------------------------------------------------------- */ + /* MDNS */ + /* -------------------------------------------------------------------------- */ + "get_peers", +]; + +fn main() { + let out_dir = std::path::PathBuf::from(std::env::var("OUT_DIR").unwrap()); + tonic_build::configure() + .file_descriptor_set_path(out_dir.join("helloworld_descriptor.bin")) + .compile(&["proto/helloworld.proto"], &["proto"]) + .expect("Failed to compile protos"); + tauri_plugin::Builder::new(COMMANDS) + .android_path("android") + .ios_path("ios") + .build(); +} diff --git a/packages/tauri-plugins/jarvis/permissions/all.toml b/packages/tauri-plugins/jarvis/permissions/all.toml new file mode 100644 index 0000000..059f06a --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/all.toml @@ -0,0 +1,92 @@ +"$schema" = "schemas/schema.json" + +[[permission]] +identifier = "allow-all" +description = "This enables all permissions for Jarvis" +commands.allow = [ + "open_devtools", + "close_devtools", + "is_devtools_open", + "toggle_devtools", + "app_is_dev", + "open_trash", + "empty_trash", + "shutdown", + "reboot", + "sleep", + "toggle_system_appearance", + "show_desktop", + "quit_all_apps", + "sleep_displays", + "set_volume", + "turn_volume_up", + "turn_volume_down", + "toggle_stage_manager", + "toggle_bluetooth", + "toggle_hidden_files", + "eject_all_disks", + "logout_user", + "toggle_mute", + "mute", + "unmute", + "hide_all_apps_except_frontmost", + "get_selected_files_in_file_explorer", + "run_apple_script", + "run_powershell", + "get_applications", + "refresh_applications_list", + "refresh_applications_list_in_bg", + # "load_manifest", + # "load_all_extensions", + "path_exists", + "start_server", + "get_server_port", + "stop_server", + "restart_server", + "set_dev_extension_folder", + "set_extension_folder", + "server_is_running", + # File System + "decompress_tarball", + "compress_tarball", + # File Search + "file_search", + "unzip", + # Path + "get_default_extensions_dir", + "get_default_extensions_storage_dir", + "is_window_label_registered", + "register_extension_window", + "unregister_extension_window", + "register_extension_spawned_process", + "unregister_extension_spawned_process", + "get_ext_label_map", + "get_frontmost_app", + # Database + "create_extension", + "get_all_extensions", + "get_unique_extension_by_identifier", + "get_unique_extension_by_path", + "get_all_extensions_by_identifier", + "delete_extension_by_path", + "delete_extension_by_ext_id", + "create_command", + "get_command_by_id", + "get_commands_by_ext_id", + "delete_command_by_id", + "update_command_by_id", + "create_extension_data", + "get_extension_data_by_id", + "search_extension_data", + "delete_extension_data_by_id", + "update_extension_data_by_id", + # Clipboard + "add_to_history", + "get_history", + # Utils + "plist_to_json", + "verify_auth", + "request_screen_capture_access", + "check_screen_capture_access", + "get_peers", +] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/add_to_history.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/add_to_history.toml new file mode 100644 index 0000000..f1d20a6 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/add_to_history.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-add-to-history" +description = "Enables the add_to_history command without any pre-configured scope." +commands.allow = ["add_to_history"] + +[[permission]] +identifier = "deny-add-to-history" +description = "Denies the add_to_history command without any pre-configured scope." +commands.deny = ["add_to_history"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/app_is_dev.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/app_is_dev.toml new file mode 100644 index 0000000..e17633b --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/app_is_dev.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-app-is-dev" +description = "Enables the app_is_dev command without any pre-configured scope." +commands.allow = ["app_is_dev"] + +[[permission]] +identifier = "deny-app-is-dev" +description = "Denies the app_is_dev command without any pre-configured scope." +commands.deny = ["app_is_dev"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/check_screen_capture_access.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/check_screen_capture_access.toml new file mode 100644 index 0000000..23d3ef6 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/check_screen_capture_access.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-check-screen-capture-access" +description = "Enables the check_screen_capture_access command without any pre-configured scope." +commands.allow = ["check_screen_capture_access"] + +[[permission]] +identifier = "deny-check-screen-capture-access" +description = "Denies the check_screen_capture_access command without any pre-configured scope." +commands.deny = ["check_screen_capture_access"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/close_devtools.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/close_devtools.toml new file mode 100644 index 0000000..b459c4b --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/close_devtools.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-close-devtools" +description = "Enables the close_devtools command without any pre-configured scope." +commands.allow = ["close_devtools"] + +[[permission]] +identifier = "deny-close-devtools" +description = "Denies the close_devtools command without any pre-configured scope." +commands.deny = ["close_devtools"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/compress_tarball.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/compress_tarball.toml new file mode 100644 index 0000000..a2f5bb9 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/compress_tarball.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-compress-tarball" +description = "Enables the compress_tarball command without any pre-configured scope." +commands.allow = ["compress_tarball"] + +[[permission]] +identifier = "deny-compress-tarball" +description = "Denies the compress_tarball command without any pre-configured scope." +commands.deny = ["compress_tarball"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_command.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_command.toml new file mode 100644 index 0000000..0cb4672 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_command.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-create-command" +description = "Enables the create_command command without any pre-configured scope." +commands.allow = ["create_command"] + +[[permission]] +identifier = "deny-create-command" +description = "Denies the create_command command without any pre-configured scope." +commands.deny = ["create_command"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_extension.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_extension.toml new file mode 100644 index 0000000..c20e75e --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_extension.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-create-extension" +description = "Enables the create_extension command without any pre-configured scope." +commands.allow = ["create_extension"] + +[[permission]] +identifier = "deny-create-extension" +description = "Denies the create_extension command without any pre-configured scope." +commands.deny = ["create_extension"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_extension_data.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_extension_data.toml new file mode 100644 index 0000000..295cbd7 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/create_extension_data.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-create-extension-data" +description = "Enables the create_extension_data command without any pre-configured scope." +commands.allow = ["create_extension_data"] + +[[permission]] +identifier = "deny-create-extension-data" +description = "Denies the create_extension_data command without any pre-configured scope." +commands.deny = ["create_extension_data"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/decompress_tarball.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/decompress_tarball.toml new file mode 100644 index 0000000..4c7e846 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/decompress_tarball.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-decompress-tarball" +description = "Enables the decompress_tarball command without any pre-configured scope." +commands.allow = ["decompress_tarball"] + +[[permission]] +identifier = "deny-decompress-tarball" +description = "Denies the decompress_tarball command without any pre-configured scope." +commands.deny = ["decompress_tarball"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_command_by_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_command_by_id.toml new file mode 100644 index 0000000..f500e5a --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_command_by_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-delete-command-by-id" +description = "Enables the delete_command_by_id command without any pre-configured scope." +commands.allow = ["delete_command_by_id"] + +[[permission]] +identifier = "deny-delete-command-by-id" +description = "Denies the delete_command_by_id command without any pre-configured scope." +commands.deny = ["delete_command_by_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_by_ext_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_by_ext_id.toml new file mode 100644 index 0000000..2e8b688 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_by_ext_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-delete-extension-by-ext-id" +description = "Enables the delete_extension_by_ext_id command without any pre-configured scope." +commands.allow = ["delete_extension_by_ext_id"] + +[[permission]] +identifier = "deny-delete-extension-by-ext-id" +description = "Denies the delete_extension_by_ext_id command without any pre-configured scope." +commands.deny = ["delete_extension_by_ext_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_by_path.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_by_path.toml new file mode 100644 index 0000000..4b05111 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_by_path.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-delete-extension-by-path" +description = "Enables the delete_extension_by_path command without any pre-configured scope." +commands.allow = ["delete_extension_by_path"] + +[[permission]] +identifier = "deny-delete-extension-by-path" +description = "Denies the delete_extension_by_path command without any pre-configured scope." +commands.deny = ["delete_extension_by_path"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_data_by_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_data_by_id.toml new file mode 100644 index 0000000..e4e2dbf --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/delete_extension_data_by_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-delete-extension-data-by-id" +description = "Enables the delete_extension_data_by_id command without any pre-configured scope." +commands.allow = ["delete_extension_data_by_id"] + +[[permission]] +identifier = "deny-delete-extension-data-by-id" +description = "Denies the delete_extension_data_by_id command without any pre-configured scope." +commands.deny = ["delete_extension_data_by_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/eject_all_disks.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/eject_all_disks.toml new file mode 100644 index 0000000..1082394 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/eject_all_disks.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-eject-all-disks" +description = "Enables the eject_all_disks command without any pre-configured scope." +commands.allow = ["eject_all_disks"] + +[[permission]] +identifier = "deny-eject-all-disks" +description = "Denies the eject_all_disks command without any pre-configured scope." +commands.deny = ["eject_all_disks"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/empty_trash.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/empty_trash.toml new file mode 100644 index 0000000..33afe4f --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/empty_trash.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-empty-trash" +description = "Enables the empty_trash command without any pre-configured scope." +commands.allow = ["empty_trash"] + +[[permission]] +identifier = "deny-empty-trash" +description = "Denies the empty_trash command without any pre-configured scope." +commands.deny = ["empty_trash"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/file_search.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/file_search.toml new file mode 100644 index 0000000..9bf9539 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/file_search.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-file-search" +description = "Enables the file_search command without any pre-configured scope." +commands.allow = ["file_search"] + +[[permission]] +identifier = "deny-file-search" +description = "Denies the file_search command without any pre-configured scope." +commands.deny = ["file_search"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_all_extensions.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_all_extensions.toml new file mode 100644 index 0000000..ca28e3a --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_all_extensions.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-all-extensions" +description = "Enables the get_all_extensions command without any pre-configured scope." +commands.allow = ["get_all_extensions"] + +[[permission]] +identifier = "deny-get-all-extensions" +description = "Denies the get_all_extensions command without any pre-configured scope." +commands.deny = ["get_all_extensions"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_all_extensions_by_identifier.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_all_extensions_by_identifier.toml new file mode 100644 index 0000000..4cd6d28 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_all_extensions_by_identifier.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-all-extensions-by-identifier" +description = "Enables the get_all_extensions_by_identifier command without any pre-configured scope." +commands.allow = ["get_all_extensions_by_identifier"] + +[[permission]] +identifier = "deny-get-all-extensions-by-identifier" +description = "Denies the get_all_extensions_by_identifier command without any pre-configured scope." +commands.deny = ["get_all_extensions_by_identifier"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_applications.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_applications.toml new file mode 100644 index 0000000..ee225b8 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_applications.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-applications" +description = "Enables the get_applications command without any pre-configured scope." +commands.allow = ["get_applications"] + +[[permission]] +identifier = "deny-get-applications" +description = "Denies the get_applications command without any pre-configured scope." +commands.deny = ["get_applications"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_command_by_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_command_by_id.toml new file mode 100644 index 0000000..2420fc0 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_command_by_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-command-by-id" +description = "Enables the get_command_by_id command without any pre-configured scope." +commands.allow = ["get_command_by_id"] + +[[permission]] +identifier = "deny-get-command-by-id" +description = "Denies the get_command_by_id command without any pre-configured scope." +commands.deny = ["get_command_by_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_commands_by_ext_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_commands_by_ext_id.toml new file mode 100644 index 0000000..9aee3ca --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_commands_by_ext_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-commands-by-ext-id" +description = "Enables the get_commands_by_ext_id command without any pre-configured scope." +commands.allow = ["get_commands_by_ext_id"] + +[[permission]] +identifier = "deny-get-commands-by-ext-id" +description = "Denies the get_commands_by_ext_id command without any pre-configured scope." +commands.deny = ["get_commands_by_ext_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_default_extensions_dir.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_default_extensions_dir.toml new file mode 100644 index 0000000..70831dd --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_default_extensions_dir.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-default-extensions-dir" +description = "Enables the get_default_extensions_dir command without any pre-configured scope." +commands.allow = ["get_default_extensions_dir"] + +[[permission]] +identifier = "deny-get-default-extensions-dir" +description = "Denies the get_default_extensions_dir command without any pre-configured scope." +commands.deny = ["get_default_extensions_dir"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_default_extensions_storage_dir.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_default_extensions_storage_dir.toml new file mode 100644 index 0000000..70bbf40 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_default_extensions_storage_dir.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-default-extensions-storage-dir" +description = "Enables the get_default_extensions_storage_dir command without any pre-configured scope." +commands.allow = ["get_default_extensions_storage_dir"] + +[[permission]] +identifier = "deny-get-default-extensions-storage-dir" +description = "Denies the get_default_extensions_storage_dir command without any pre-configured scope." +commands.deny = ["get_default_extensions_storage_dir"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_dev_extension_folder.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_dev_extension_folder.toml new file mode 100644 index 0000000..22a72bd --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_dev_extension_folder.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-dev-extension-folder" +description = "Enables the get_dev_extension_folder command without any pre-configured scope." +commands.allow = ["get_dev_extension_folder"] + +[[permission]] +identifier = "deny-get-dev-extension-folder" +description = "Denies the get_dev_extension_folder command without any pre-configured scope." +commands.deny = ["get_dev_extension_folder"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_ext_label_map.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_ext_label_map.toml new file mode 100644 index 0000000..6b242cf --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_ext_label_map.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-ext-label-map" +description = "Enables the get_ext_label_map command without any pre-configured scope." +commands.allow = ["get_ext_label_map"] + +[[permission]] +identifier = "deny-get-ext-label-map" +description = "Denies the get_ext_label_map command without any pre-configured scope." +commands.deny = ["get_ext_label_map"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_extension_data_by_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_extension_data_by_id.toml new file mode 100644 index 0000000..34bcf25 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_extension_data_by_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-extension-data-by-id" +description = "Enables the get_extension_data_by_id command without any pre-configured scope." +commands.allow = ["get_extension_data_by_id"] + +[[permission]] +identifier = "deny-get-extension-data-by-id" +description = "Denies the get_extension_data_by_id command without any pre-configured scope." +commands.deny = ["get_extension_data_by_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_extension_folder.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_extension_folder.toml new file mode 100644 index 0000000..b16fa76 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_extension_folder.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-extension-folder" +description = "Enables the get_extension_folder command without any pre-configured scope." +commands.allow = ["get_extension_folder"] + +[[permission]] +identifier = "deny-get-extension-folder" +description = "Denies the get_extension_folder command without any pre-configured scope." +commands.deny = ["get_extension_folder"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_history.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_history.toml new file mode 100644 index 0000000..14c61fa --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_history.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-history" +description = "Enables the get_history command without any pre-configured scope." +commands.allow = ["get_history"] + +[[permission]] +identifier = "deny-get-history" +description = "Denies the get_history command without any pre-configured scope." +commands.deny = ["get_history"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_peers.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_peers.toml new file mode 100644 index 0000000..ca8f68b --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_peers.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-peers" +description = "Enables the get_peers command without any pre-configured scope." +commands.allow = ["get_peers"] + +[[permission]] +identifier = "deny-get-peers" +description = "Denies the get_peers command without any pre-configured scope." +commands.deny = ["get_peers"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_selected_files_in_file_explorer.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_selected_files_in_file_explorer.toml new file mode 100644 index 0000000..a30f2e0 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_selected_files_in_file_explorer.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-selected-files-in-file-explorer" +description = "Enables the get_selected_files_in_file_explorer command without any pre-configured scope." +commands.allow = ["get_selected_files_in_file_explorer"] + +[[permission]] +identifier = "deny-get-selected-files-in-file-explorer" +description = "Denies the get_selected_files_in_file_explorer command without any pre-configured scope." +commands.deny = ["get_selected_files_in_file_explorer"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_server_port.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_server_port.toml new file mode 100644 index 0000000..c36a8f1 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_server_port.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-server-port" +description = "Enables the get_server_port command without any pre-configured scope." +commands.allow = ["get_server_port"] + +[[permission]] +identifier = "deny-get-server-port" +description = "Denies the get_server_port command without any pre-configured scope." +commands.deny = ["get_server_port"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_unique_extension_by_identifier.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_unique_extension_by_identifier.toml new file mode 100644 index 0000000..fd2b60f --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_unique_extension_by_identifier.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-unique-extension-by-identifier" +description = "Enables the get_unique_extension_by_identifier command without any pre-configured scope." +commands.allow = ["get_unique_extension_by_identifier"] + +[[permission]] +identifier = "deny-get-unique-extension-by-identifier" +description = "Denies the get_unique_extension_by_identifier command without any pre-configured scope." +commands.deny = ["get_unique_extension_by_identifier"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_unique_extension_by_path.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_unique_extension_by_path.toml new file mode 100644 index 0000000..36e91a0 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/get_unique_extension_by_path.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-get-unique-extension-by-path" +description = "Enables the get_unique_extension_by_path command without any pre-configured scope." +commands.allow = ["get_unique_extension_by_path"] + +[[permission]] +identifier = "deny-get-unique-extension-by-path" +description = "Denies the get_unique_extension_by_path command without any pre-configured scope." +commands.deny = ["get_unique_extension_by_path"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/hide_all_apps_except_frontmost.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/hide_all_apps_except_frontmost.toml new file mode 100644 index 0000000..2e2aa96 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/hide_all_apps_except_frontmost.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-hide-all-apps-except-frontmost" +description = "Enables the hide_all_apps_except_frontmost command without any pre-configured scope." +commands.allow = ["hide_all_apps_except_frontmost"] + +[[permission]] +identifier = "deny-hide-all-apps-except-frontmost" +description = "Denies the hide_all_apps_except_frontmost command without any pre-configured scope." +commands.deny = ["hide_all_apps_except_frontmost"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/is_devtools_open.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/is_devtools_open.toml new file mode 100644 index 0000000..1b44c80 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/is_devtools_open.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-is-devtools-open" +description = "Enables the is_devtools_open command without any pre-configured scope." +commands.allow = ["is_devtools_open"] + +[[permission]] +identifier = "deny-is-devtools-open" +description = "Denies the is_devtools_open command without any pre-configured scope." +commands.deny = ["is_devtools_open"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/is_window_label_registered.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/is_window_label_registered.toml new file mode 100644 index 0000000..6bf1d44 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/is_window_label_registered.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-is-window-label-registered" +description = "Enables the is_window_label_registered command without any pre-configured scope." +commands.allow = ["is_window_label_registered"] + +[[permission]] +identifier = "deny-is-window-label-registered" +description = "Denies the is_window_label_registered command without any pre-configured scope." +commands.deny = ["is_window_label_registered"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/logout_user.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/logout_user.toml new file mode 100644 index 0000000..7136c52 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/logout_user.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-logout-user" +description = "Enables the logout_user command without any pre-configured scope." +commands.allow = ["logout_user"] + +[[permission]] +identifier = "deny-logout-user" +description = "Denies the logout_user command without any pre-configured scope." +commands.deny = ["logout_user"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/mute.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/mute.toml new file mode 100644 index 0000000..5e589b9 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/mute.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-mute" +description = "Enables the mute command without any pre-configured scope." +commands.allow = ["mute"] + +[[permission]] +identifier = "deny-mute" +description = "Denies the mute command without any pre-configured scope." +commands.deny = ["mute"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/open_devtools.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/open_devtools.toml new file mode 100644 index 0000000..e22581e --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/open_devtools.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-open-devtools" +description = "Enables the open_devtools command without any pre-configured scope." +commands.allow = ["open_devtools"] + +[[permission]] +identifier = "deny-open-devtools" +description = "Denies the open_devtools command without any pre-configured scope." +commands.deny = ["open_devtools"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/open_trash.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/open_trash.toml new file mode 100644 index 0000000..4e1c647 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/open_trash.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-open-trash" +description = "Enables the open_trash command without any pre-configured scope." +commands.allow = ["open_trash"] + +[[permission]] +identifier = "deny-open-trash" +description = "Denies the open_trash command without any pre-configured scope." +commands.deny = ["open_trash"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/path_exists.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/path_exists.toml new file mode 100644 index 0000000..6308ed6 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/path_exists.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-path-exists" +description = "Enables the path_exists command without any pre-configured scope." +commands.allow = ["path_exists"] + +[[permission]] +identifier = "deny-path-exists" +description = "Denies the path_exists command without any pre-configured scope." +commands.deny = ["path_exists"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/plist_to_json.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/plist_to_json.toml new file mode 100644 index 0000000..220379e --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/plist_to_json.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-plist-to-json" +description = "Enables the plist_to_json command without any pre-configured scope." +commands.allow = ["plist_to_json"] + +[[permission]] +identifier = "deny-plist-to-json" +description = "Denies the plist_to_json command without any pre-configured scope." +commands.deny = ["plist_to_json"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/quit_all_apps.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/quit_all_apps.toml new file mode 100644 index 0000000..479a6bf --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/quit_all_apps.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-quit-all-apps" +description = "Enables the quit_all_apps command without any pre-configured scope." +commands.allow = ["quit_all_apps"] + +[[permission]] +identifier = "deny-quit-all-apps" +description = "Denies the quit_all_apps command without any pre-configured scope." +commands.deny = ["quit_all_apps"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/reboot.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/reboot.toml new file mode 100644 index 0000000..536b134 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/reboot.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-reboot" +description = "Enables the reboot command without any pre-configured scope." +commands.allow = ["reboot"] + +[[permission]] +identifier = "deny-reboot" +description = "Denies the reboot command without any pre-configured scope." +commands.deny = ["reboot"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/refresh_applications_list.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/refresh_applications_list.toml new file mode 100644 index 0000000..ef51af6 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/refresh_applications_list.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-refresh-applications-list" +description = "Enables the refresh_applications_list command without any pre-configured scope." +commands.allow = ["refresh_applications_list"] + +[[permission]] +identifier = "deny-refresh-applications-list" +description = "Denies the refresh_applications_list command without any pre-configured scope." +commands.deny = ["refresh_applications_list"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/refresh_applications_list_in_bg.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/refresh_applications_list_in_bg.toml new file mode 100644 index 0000000..77387de --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/refresh_applications_list_in_bg.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-refresh-applications-list-in-bg" +description = "Enables the refresh_applications_list_in_bg command without any pre-configured scope." +commands.allow = ["refresh_applications_list_in_bg"] + +[[permission]] +identifier = "deny-refresh-applications-list-in-bg" +description = "Denies the refresh_applications_list_in_bg command without any pre-configured scope." +commands.deny = ["refresh_applications_list_in_bg"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/register_extension_spawned_process.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/register_extension_spawned_process.toml new file mode 100644 index 0000000..41d039b --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/register_extension_spawned_process.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-register-extension-spawned-process" +description = "Enables the register_extension_spawned_process command without any pre-configured scope." +commands.allow = ["register_extension_spawned_process"] + +[[permission]] +identifier = "deny-register-extension-spawned-process" +description = "Denies the register_extension_spawned_process command without any pre-configured scope." +commands.deny = ["register_extension_spawned_process"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/register_extension_window.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/register_extension_window.toml new file mode 100644 index 0000000..4869f8b --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/register_extension_window.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-register-extension-window" +description = "Enables the register_extension_window command without any pre-configured scope." +commands.allow = ["register_extension_window"] + +[[permission]] +identifier = "deny-register-extension-window" +description = "Denies the register_extension_window command without any pre-configured scope." +commands.deny = ["register_extension_window"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/request_screen_capture_access.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/request_screen_capture_access.toml new file mode 100644 index 0000000..75463f7 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/request_screen_capture_access.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-request-screen-capture-access" +description = "Enables the request_screen_capture_access command without any pre-configured scope." +commands.allow = ["request_screen_capture_access"] + +[[permission]] +identifier = "deny-request-screen-capture-access" +description = "Denies the request_screen_capture_access command without any pre-configured scope." +commands.deny = ["request_screen_capture_access"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/restart_server.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/restart_server.toml new file mode 100644 index 0000000..e3712dc --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/restart_server.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-restart-server" +description = "Enables the restart_server command without any pre-configured scope." +commands.allow = ["restart_server"] + +[[permission]] +identifier = "deny-restart-server" +description = "Denies the restart_server command without any pre-configured scope." +commands.deny = ["restart_server"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/run_apple_script.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/run_apple_script.toml new file mode 100644 index 0000000..e85dea9 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/run_apple_script.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-run-apple-script" +description = "Enables the run_apple_script command without any pre-configured scope." +commands.allow = ["run_apple_script"] + +[[permission]] +identifier = "deny-run-apple-script" +description = "Denies the run_apple_script command without any pre-configured scope." +commands.deny = ["run_apple_script"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/run_powershell.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/run_powershell.toml new file mode 100644 index 0000000..eafa142 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/run_powershell.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-run-powershell" +description = "Enables the run_powershell command without any pre-configured scope." +commands.allow = ["run_powershell"] + +[[permission]] +identifier = "deny-run-powershell" +description = "Denies the run_powershell command without any pre-configured scope." +commands.deny = ["run_powershell"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/search_extension_data.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/search_extension_data.toml new file mode 100644 index 0000000..5a73a67 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/search_extension_data.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-search-extension-data" +description = "Enables the search_extension_data command without any pre-configured scope." +commands.allow = ["search_extension_data"] + +[[permission]] +identifier = "deny-search-extension-data" +description = "Denies the search_extension_data command without any pre-configured scope." +commands.deny = ["search_extension_data"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/server_is_running.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/server_is_running.toml new file mode 100644 index 0000000..4f0dc59 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/server_is_running.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-server-is-running" +description = "Enables the server_is_running command without any pre-configured scope." +commands.allow = ["server_is_running"] + +[[permission]] +identifier = "deny-server-is-running" +description = "Denies the server_is_running command without any pre-configured scope." +commands.deny = ["server_is_running"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_dev_extension_folder.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_dev_extension_folder.toml new file mode 100644 index 0000000..ee89d45 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_dev_extension_folder.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-set-dev-extension-folder" +description = "Enables the set_dev_extension_folder command without any pre-configured scope." +commands.allow = ["set_dev_extension_folder"] + +[[permission]] +identifier = "deny-set-dev-extension-folder" +description = "Denies the set_dev_extension_folder command without any pre-configured scope." +commands.deny = ["set_dev_extension_folder"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_extension_folder.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_extension_folder.toml new file mode 100644 index 0000000..d46390b --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_extension_folder.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-set-extension-folder" +description = "Enables the set_extension_folder command without any pre-configured scope." +commands.allow = ["set_extension_folder"] + +[[permission]] +identifier = "deny-set-extension-folder" +description = "Denies the set_extension_folder command without any pre-configured scope." +commands.deny = ["set_extension_folder"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_volume.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_volume.toml new file mode 100644 index 0000000..8c6d2db --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/set_volume.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-set-volume" +description = "Enables the set_volume command without any pre-configured scope." +commands.allow = ["set_volume"] + +[[permission]] +identifier = "deny-set-volume" +description = "Denies the set_volume command without any pre-configured scope." +commands.deny = ["set_volume"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/show_desktop.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/show_desktop.toml new file mode 100644 index 0000000..5c6f0ca --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/show_desktop.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-show-desktop" +description = "Enables the show_desktop command without any pre-configured scope." +commands.allow = ["show_desktop"] + +[[permission]] +identifier = "deny-show-desktop" +description = "Denies the show_desktop command without any pre-configured scope." +commands.deny = ["show_desktop"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/shutdown.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/shutdown.toml new file mode 100644 index 0000000..6a0a329 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/shutdown.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-shutdown" +description = "Enables the shutdown command without any pre-configured scope." +commands.allow = ["shutdown"] + +[[permission]] +identifier = "deny-shutdown" +description = "Denies the shutdown command without any pre-configured scope." +commands.deny = ["shutdown"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/sleep.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/sleep.toml new file mode 100644 index 0000000..1240596 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/sleep.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-sleep" +description = "Enables the sleep command without any pre-configured scope." +commands.allow = ["sleep"] + +[[permission]] +identifier = "deny-sleep" +description = "Denies the sleep command without any pre-configured scope." +commands.deny = ["sleep"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/sleep_displays.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/sleep_displays.toml new file mode 100644 index 0000000..254092b --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/sleep_displays.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-sleep-displays" +description = "Enables the sleep_displays command without any pre-configured scope." +commands.allow = ["sleep_displays"] + +[[permission]] +identifier = "deny-sleep-displays" +description = "Denies the sleep_displays command without any pre-configured scope." +commands.deny = ["sleep_displays"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/start_server.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/start_server.toml new file mode 100644 index 0000000..ebd7332 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/start_server.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-start-server" +description = "Enables the start_server command without any pre-configured scope." +commands.allow = ["start_server"] + +[[permission]] +identifier = "deny-start-server" +description = "Denies the start_server command without any pre-configured scope." +commands.deny = ["start_server"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/stop_server.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/stop_server.toml new file mode 100644 index 0000000..8a98ff8 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/stop_server.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-stop-server" +description = "Enables the stop_server command without any pre-configured scope." +commands.allow = ["stop_server"] + +[[permission]] +identifier = "deny-stop-server" +description = "Denies the stop_server command without any pre-configured scope." +commands.deny = ["stop_server"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_bluetooth.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_bluetooth.toml new file mode 100644 index 0000000..8a235ab --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_bluetooth.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-toggle-bluetooth" +description = "Enables the toggle_bluetooth command without any pre-configured scope." +commands.allow = ["toggle_bluetooth"] + +[[permission]] +identifier = "deny-toggle-bluetooth" +description = "Denies the toggle_bluetooth command without any pre-configured scope." +commands.deny = ["toggle_bluetooth"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_devtools.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_devtools.toml new file mode 100644 index 0000000..40049b7 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_devtools.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-toggle-devtools" +description = "Enables the toggle_devtools command without any pre-configured scope." +commands.allow = ["toggle_devtools"] + +[[permission]] +identifier = "deny-toggle-devtools" +description = "Denies the toggle_devtools command without any pre-configured scope." +commands.deny = ["toggle_devtools"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_hidden_files.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_hidden_files.toml new file mode 100644 index 0000000..618a228 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_hidden_files.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-toggle-hidden-files" +description = "Enables the toggle_hidden_files command without any pre-configured scope." +commands.allow = ["toggle_hidden_files"] + +[[permission]] +identifier = "deny-toggle-hidden-files" +description = "Denies the toggle_hidden_files command without any pre-configured scope." +commands.deny = ["toggle_hidden_files"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_mute.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_mute.toml new file mode 100644 index 0000000..73bdd4e --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_mute.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-toggle-mute" +description = "Enables the toggle_mute command without any pre-configured scope." +commands.allow = ["toggle_mute"] + +[[permission]] +identifier = "deny-toggle-mute" +description = "Denies the toggle_mute command without any pre-configured scope." +commands.deny = ["toggle_mute"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_stage_manager.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_stage_manager.toml new file mode 100644 index 0000000..ed11762 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_stage_manager.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-toggle-stage-manager" +description = "Enables the toggle_stage_manager command without any pre-configured scope." +commands.allow = ["toggle_stage_manager"] + +[[permission]] +identifier = "deny-toggle-stage-manager" +description = "Denies the toggle_stage_manager command without any pre-configured scope." +commands.deny = ["toggle_stage_manager"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_system_appearance.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_system_appearance.toml new file mode 100644 index 0000000..9684f00 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/toggle_system_appearance.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-toggle-system-appearance" +description = "Enables the toggle_system_appearance command without any pre-configured scope." +commands.allow = ["toggle_system_appearance"] + +[[permission]] +identifier = "deny-toggle-system-appearance" +description = "Denies the toggle_system_appearance command without any pre-configured scope." +commands.deny = ["toggle_system_appearance"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/turn_volume_down.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/turn_volume_down.toml new file mode 100644 index 0000000..8c614af --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/turn_volume_down.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-turn-volume-down" +description = "Enables the turn_volume_down command without any pre-configured scope." +commands.allow = ["turn_volume_down"] + +[[permission]] +identifier = "deny-turn-volume-down" +description = "Denies the turn_volume_down command without any pre-configured scope." +commands.deny = ["turn_volume_down"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/turn_volume_up.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/turn_volume_up.toml new file mode 100644 index 0000000..f02e439 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/turn_volume_up.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-turn-volume-up" +description = "Enables the turn_volume_up command without any pre-configured scope." +commands.allow = ["turn_volume_up"] + +[[permission]] +identifier = "deny-turn-volume-up" +description = "Denies the turn_volume_up command without any pre-configured scope." +commands.deny = ["turn_volume_up"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unmute.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unmute.toml new file mode 100644 index 0000000..723bd99 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unmute.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-unmute" +description = "Enables the unmute command without any pre-configured scope." +commands.allow = ["unmute"] + +[[permission]] +identifier = "deny-unmute" +description = "Denies the unmute command without any pre-configured scope." +commands.deny = ["unmute"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unregister_extension_spawned_process.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unregister_extension_spawned_process.toml new file mode 100644 index 0000000..93bbc01 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unregister_extension_spawned_process.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-unregister-extension-spawned-process" +description = "Enables the unregister_extension_spawned_process command without any pre-configured scope." +commands.allow = ["unregister_extension_spawned_process"] + +[[permission]] +identifier = "deny-unregister-extension-spawned-process" +description = "Denies the unregister_extension_spawned_process command without any pre-configured scope." +commands.deny = ["unregister_extension_spawned_process"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unregister_extension_window.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unregister_extension_window.toml new file mode 100644 index 0000000..e5245a7 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unregister_extension_window.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-unregister-extension-window" +description = "Enables the unregister_extension_window command without any pre-configured scope." +commands.allow = ["unregister_extension_window"] + +[[permission]] +identifier = "deny-unregister-extension-window" +description = "Denies the unregister_extension_window command without any pre-configured scope." +commands.deny = ["unregister_extension_window"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unzip.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unzip.toml new file mode 100644 index 0000000..d17566c --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/unzip.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-unzip" +description = "Enables the unzip command without any pre-configured scope." +commands.allow = ["unzip"] + +[[permission]] +identifier = "deny-unzip" +description = "Denies the unzip command without any pre-configured scope." +commands.deny = ["unzip"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/update_command_by_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/update_command_by_id.toml new file mode 100644 index 0000000..037ac95 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/update_command_by_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-update-command-by-id" +description = "Enables the update_command_by_id command without any pre-configured scope." +commands.allow = ["update_command_by_id"] + +[[permission]] +identifier = "deny-update-command-by-id" +description = "Denies the update_command_by_id command without any pre-configured scope." +commands.deny = ["update_command_by_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/update_extension_data_by_id.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/update_extension_data_by_id.toml new file mode 100644 index 0000000..2886cef --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/update_extension_data_by_id.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-update-extension-data-by-id" +description = "Enables the update_extension_data_by_id command without any pre-configured scope." +commands.allow = ["update_extension_data_by_id"] + +[[permission]] +identifier = "deny-update-extension-data-by-id" +description = "Denies the update_extension_data_by_id command without any pre-configured scope." +commands.deny = ["update_extension_data_by_id"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/verify_auth.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/verify_auth.toml new file mode 100644 index 0000000..f0f91e4 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/verify_auth.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-verify-auth" +description = "Enables the verify_auth command without any pre-configured scope." +commands.allow = ["verify_auth"] + +[[permission]] +identifier = "deny-verify-auth" +description = "Denies the verify_auth command without any pre-configured scope." +commands.deny = ["verify_auth"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md b/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md new file mode 100644 index 0000000..c99f91c --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md @@ -0,0 +1,2062 @@ +## Permission Table + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
IdentifierDescription
+ +`jarvis:allow-all` + + + +This enables all permissions for Jarvis + +
+ +`jarvis:allow-add-to-history` + + + +Enables the add_to_history command without any pre-configured scope. + +
+ +`jarvis:deny-add-to-history` + + + +Denies the add_to_history command without any pre-configured scope. + +
+ +`jarvis:allow-app-is-dev` + + + +Enables the app_is_dev command without any pre-configured scope. + +
+ +`jarvis:deny-app-is-dev` + + + +Denies the app_is_dev command without any pre-configured scope. + +
+ +`jarvis:allow-check-screen-capture-access` + + + +Enables the check_screen_capture_access command without any pre-configured scope. + +
+ +`jarvis:deny-check-screen-capture-access` + + + +Denies the check_screen_capture_access command without any pre-configured scope. + +
+ +`jarvis:allow-close-devtools` + + + +Enables the close_devtools command without any pre-configured scope. + +
+ +`jarvis:deny-close-devtools` + + + +Denies the close_devtools command without any pre-configured scope. + +
+ +`jarvis:allow-compress-tarball` + + + +Enables the compress_tarball command without any pre-configured scope. + +
+ +`jarvis:deny-compress-tarball` + + + +Denies the compress_tarball command without any pre-configured scope. + +
+ +`jarvis:allow-create-command` + + + +Enables the create_command command without any pre-configured scope. + +
+ +`jarvis:deny-create-command` + + + +Denies the create_command command without any pre-configured scope. + +
+ +`jarvis:allow-create-extension` + + + +Enables the create_extension command without any pre-configured scope. + +
+ +`jarvis:deny-create-extension` + + + +Denies the create_extension command without any pre-configured scope. + +
+ +`jarvis:allow-create-extension-data` + + + +Enables the create_extension_data command without any pre-configured scope. + +
+ +`jarvis:deny-create-extension-data` + + + +Denies the create_extension_data command without any pre-configured scope. + +
+ +`jarvis:allow-decompress-tarball` + + + +Enables the decompress_tarball command without any pre-configured scope. + +
+ +`jarvis:deny-decompress-tarball` + + + +Denies the decompress_tarball command without any pre-configured scope. + +
+ +`jarvis:allow-delete-command-by-id` + + + +Enables the delete_command_by_id command without any pre-configured scope. + +
+ +`jarvis:deny-delete-command-by-id` + + + +Denies the delete_command_by_id command without any pre-configured scope. + +
+ +`jarvis:allow-delete-extension-by-ext-id` + + + +Enables the delete_extension_by_ext_id command without any pre-configured scope. + +
+ +`jarvis:deny-delete-extension-by-ext-id` + + + +Denies the delete_extension_by_ext_id command without any pre-configured scope. + +
+ +`jarvis:allow-delete-extension-by-path` + + + +Enables the delete_extension_by_path command without any pre-configured scope. + +
+ +`jarvis:deny-delete-extension-by-path` + + + +Denies the delete_extension_by_path command without any pre-configured scope. + +
+ +`jarvis:allow-delete-extension-data-by-id` + + + +Enables the delete_extension_data_by_id command without any pre-configured scope. + +
+ +`jarvis:deny-delete-extension-data-by-id` + + + +Denies the delete_extension_data_by_id command without any pre-configured scope. + +
+ +`jarvis:allow-eject-all-disks` + + + +Enables the eject_all_disks command without any pre-configured scope. + +
+ +`jarvis:deny-eject-all-disks` + + + +Denies the eject_all_disks command without any pre-configured scope. + +
+ +`jarvis:allow-empty-trash` + + + +Enables the empty_trash command without any pre-configured scope. + +
+ +`jarvis:deny-empty-trash` + + + +Denies the empty_trash command without any pre-configured scope. + +
+ +`jarvis:allow-file-search` + + + +Enables the file_search command without any pre-configured scope. + +
+ +`jarvis:deny-file-search` + + + +Denies the file_search command without any pre-configured scope. + +
+ +`jarvis:allow-get-all-extensions` + + + +Enables the get_all_extensions command without any pre-configured scope. + +
+ +`jarvis:deny-get-all-extensions` + + + +Denies the get_all_extensions command without any pre-configured scope. + +
+ +`jarvis:allow-get-all-extensions-by-identifier` + + + +Enables the get_all_extensions_by_identifier command without any pre-configured scope. + +
+ +`jarvis:deny-get-all-extensions-by-identifier` + + + +Denies the get_all_extensions_by_identifier command without any pre-configured scope. + +
+ +`jarvis:allow-get-applications` + + + +Enables the get_applications command without any pre-configured scope. + +
+ +`jarvis:deny-get-applications` + + + +Denies the get_applications command without any pre-configured scope. + +
+ +`jarvis:allow-get-command-by-id` + + + +Enables the get_command_by_id command without any pre-configured scope. + +
+ +`jarvis:deny-get-command-by-id` + + + +Denies the get_command_by_id command without any pre-configured scope. + +
+ +`jarvis:allow-get-commands-by-ext-id` + + + +Enables the get_commands_by_ext_id command without any pre-configured scope. + +
+ +`jarvis:deny-get-commands-by-ext-id` + + + +Denies the get_commands_by_ext_id command without any pre-configured scope. + +
+ +`jarvis:allow-get-default-extensions-dir` + + + +Enables the get_default_extensions_dir command without any pre-configured scope. + +
+ +`jarvis:deny-get-default-extensions-dir` + + + +Denies the get_default_extensions_dir command without any pre-configured scope. + +
+ +`jarvis:allow-get-default-extensions-storage-dir` + + + +Enables the get_default_extensions_storage_dir command without any pre-configured scope. + +
+ +`jarvis:deny-get-default-extensions-storage-dir` + + + +Denies the get_default_extensions_storage_dir command without any pre-configured scope. + +
+ +`jarvis:allow-get-dev-extension-folder` + + + +Enables the get_dev_extension_folder command without any pre-configured scope. + +
+ +`jarvis:deny-get-dev-extension-folder` + + + +Denies the get_dev_extension_folder command without any pre-configured scope. + +
+ +`jarvis:allow-get-ext-label-map` + + + +Enables the get_ext_label_map command without any pre-configured scope. + +
+ +`jarvis:deny-get-ext-label-map` + + + +Denies the get_ext_label_map command without any pre-configured scope. + +
+ +`jarvis:allow-get-extension-data-by-id` + + + +Enables the get_extension_data_by_id command without any pre-configured scope. + +
+ +`jarvis:deny-get-extension-data-by-id` + + + +Denies the get_extension_data_by_id command without any pre-configured scope. + +
+ +`jarvis:allow-get-extension-folder` + + + +Enables the get_extension_folder command without any pre-configured scope. + +
+ +`jarvis:deny-get-extension-folder` + + + +Denies the get_extension_folder command without any pre-configured scope. + +
+ +`jarvis:allow-get-history` + + + +Enables the get_history command without any pre-configured scope. + +
+ +`jarvis:deny-get-history` + + + +Denies the get_history command without any pre-configured scope. + +
+ +`jarvis:allow-get-peers` + + + +Enables the get_peers command without any pre-configured scope. + +
+ +`jarvis:deny-get-peers` + + + +Denies the get_peers command without any pre-configured scope. + +
+ +`jarvis:allow-get-selected-files-in-file-explorer` + + + +Enables the get_selected_files_in_file_explorer command without any pre-configured scope. + +
+ +`jarvis:deny-get-selected-files-in-file-explorer` + + + +Denies the get_selected_files_in_file_explorer command without any pre-configured scope. + +
+ +`jarvis:allow-get-server-port` + + + +Enables the get_server_port command without any pre-configured scope. + +
+ +`jarvis:deny-get-server-port` + + + +Denies the get_server_port command without any pre-configured scope. + +
+ +`jarvis:allow-get-unique-extension-by-identifier` + + + +Enables the get_unique_extension_by_identifier command without any pre-configured scope. + +
+ +`jarvis:deny-get-unique-extension-by-identifier` + + + +Denies the get_unique_extension_by_identifier command without any pre-configured scope. + +
+ +`jarvis:allow-get-unique-extension-by-path` + + + +Enables the get_unique_extension_by_path command without any pre-configured scope. + +
+ +`jarvis:deny-get-unique-extension-by-path` + + + +Denies the get_unique_extension_by_path command without any pre-configured scope. + +
+ +`jarvis:allow-hide-all-apps-except-frontmost` + + + +Enables the hide_all_apps_except_frontmost command without any pre-configured scope. + +
+ +`jarvis:deny-hide-all-apps-except-frontmost` + + + +Denies the hide_all_apps_except_frontmost command without any pre-configured scope. + +
+ +`jarvis:allow-is-devtools-open` + + + +Enables the is_devtools_open command without any pre-configured scope. + +
+ +`jarvis:deny-is-devtools-open` + + + +Denies the is_devtools_open command without any pre-configured scope. + +
+ +`jarvis:allow-is-window-label-registered` + + + +Enables the is_window_label_registered command without any pre-configured scope. + +
+ +`jarvis:deny-is-window-label-registered` + + + +Denies the is_window_label_registered command without any pre-configured scope. + +
+ +`jarvis:allow-logout-user` + + + +Enables the logout_user command without any pre-configured scope. + +
+ +`jarvis:deny-logout-user` + + + +Denies the logout_user command without any pre-configured scope. + +
+ +`jarvis:allow-mute` + + + +Enables the mute command without any pre-configured scope. + +
+ +`jarvis:deny-mute` + + + +Denies the mute command without any pre-configured scope. + +
+ +`jarvis:allow-open-devtools` + + + +Enables the open_devtools command without any pre-configured scope. + +
+ +`jarvis:deny-open-devtools` + + + +Denies the open_devtools command without any pre-configured scope. + +
+ +`jarvis:allow-open-trash` + + + +Enables the open_trash command without any pre-configured scope. + +
+ +`jarvis:deny-open-trash` + + + +Denies the open_trash command without any pre-configured scope. + +
+ +`jarvis:allow-path-exists` + + + +Enables the path_exists command without any pre-configured scope. + +
+ +`jarvis:deny-path-exists` + + + +Denies the path_exists command without any pre-configured scope. + +
+ +`jarvis:allow-plist-to-json` + + + +Enables the plist_to_json command without any pre-configured scope. + +
+ +`jarvis:deny-plist-to-json` + + + +Denies the plist_to_json command without any pre-configured scope. + +
+ +`jarvis:allow-quit-all-apps` + + + +Enables the quit_all_apps command without any pre-configured scope. + +
+ +`jarvis:deny-quit-all-apps` + + + +Denies the quit_all_apps command without any pre-configured scope. + +
+ +`jarvis:allow-reboot` + + + +Enables the reboot command without any pre-configured scope. + +
+ +`jarvis:deny-reboot` + + + +Denies the reboot command without any pre-configured scope. + +
+ +`jarvis:allow-refresh-applications-list` + + + +Enables the refresh_applications_list command without any pre-configured scope. + +
+ +`jarvis:deny-refresh-applications-list` + + + +Denies the refresh_applications_list command without any pre-configured scope. + +
+ +`jarvis:allow-refresh-applications-list-in-bg` + + + +Enables the refresh_applications_list_in_bg command without any pre-configured scope. + +
+ +`jarvis:deny-refresh-applications-list-in-bg` + + + +Denies the refresh_applications_list_in_bg command without any pre-configured scope. + +
+ +`jarvis:allow-register-extension-spawned-process` + + + +Enables the register_extension_spawned_process command without any pre-configured scope. + +
+ +`jarvis:deny-register-extension-spawned-process` + + + +Denies the register_extension_spawned_process command without any pre-configured scope. + +
+ +`jarvis:allow-register-extension-window` + + + +Enables the register_extension_window command without any pre-configured scope. + +
+ +`jarvis:deny-register-extension-window` + + + +Denies the register_extension_window command without any pre-configured scope. + +
+ +`jarvis:allow-request-screen-capture-access` + + + +Enables the request_screen_capture_access command without any pre-configured scope. + +
+ +`jarvis:deny-request-screen-capture-access` + + + +Denies the request_screen_capture_access command without any pre-configured scope. + +
+ +`jarvis:allow-restart-server` + + + +Enables the restart_server command without any pre-configured scope. + +
+ +`jarvis:deny-restart-server` + + + +Denies the restart_server command without any pre-configured scope. + +
+ +`jarvis:allow-run-apple-script` + + + +Enables the run_apple_script command without any pre-configured scope. + +
+ +`jarvis:deny-run-apple-script` + + + +Denies the run_apple_script command without any pre-configured scope. + +
+ +`jarvis:allow-run-powershell` + + + +Enables the run_powershell command without any pre-configured scope. + +
+ +`jarvis:deny-run-powershell` + + + +Denies the run_powershell command without any pre-configured scope. + +
+ +`jarvis:allow-search-extension-data` + + + +Enables the search_extension_data command without any pre-configured scope. + +
+ +`jarvis:deny-search-extension-data` + + + +Denies the search_extension_data command without any pre-configured scope. + +
+ +`jarvis:allow-server-is-running` + + + +Enables the server_is_running command without any pre-configured scope. + +
+ +`jarvis:deny-server-is-running` + + + +Denies the server_is_running command without any pre-configured scope. + +
+ +`jarvis:allow-set-dev-extension-folder` + + + +Enables the set_dev_extension_folder command without any pre-configured scope. + +
+ +`jarvis:deny-set-dev-extension-folder` + + + +Denies the set_dev_extension_folder command without any pre-configured scope. + +
+ +`jarvis:allow-set-extension-folder` + + + +Enables the set_extension_folder command without any pre-configured scope. + +
+ +`jarvis:deny-set-extension-folder` + + + +Denies the set_extension_folder command without any pre-configured scope. + +
+ +`jarvis:allow-set-volume` + + + +Enables the set_volume command without any pre-configured scope. + +
+ +`jarvis:deny-set-volume` + + + +Denies the set_volume command without any pre-configured scope. + +
+ +`jarvis:allow-show-desktop` + + + +Enables the show_desktop command without any pre-configured scope. + +
+ +`jarvis:deny-show-desktop` + + + +Denies the show_desktop command without any pre-configured scope. + +
+ +`jarvis:allow-shutdown` + + + +Enables the shutdown command without any pre-configured scope. + +
+ +`jarvis:deny-shutdown` + + + +Denies the shutdown command without any pre-configured scope. + +
+ +`jarvis:allow-sleep` + + + +Enables the sleep command without any pre-configured scope. + +
+ +`jarvis:deny-sleep` + + + +Denies the sleep command without any pre-configured scope. + +
+ +`jarvis:allow-sleep-displays` + + + +Enables the sleep_displays command without any pre-configured scope. + +
+ +`jarvis:deny-sleep-displays` + + + +Denies the sleep_displays command without any pre-configured scope. + +
+ +`jarvis:allow-start-server` + + + +Enables the start_server command without any pre-configured scope. + +
+ +`jarvis:deny-start-server` + + + +Denies the start_server command without any pre-configured scope. + +
+ +`jarvis:allow-stop-server` + + + +Enables the stop_server command without any pre-configured scope. + +
+ +`jarvis:deny-stop-server` + + + +Denies the stop_server command without any pre-configured scope. + +
+ +`jarvis:allow-toggle-bluetooth` + + + +Enables the toggle_bluetooth command without any pre-configured scope. + +
+ +`jarvis:deny-toggle-bluetooth` + + + +Denies the toggle_bluetooth command without any pre-configured scope. + +
+ +`jarvis:allow-toggle-devtools` + + + +Enables the toggle_devtools command without any pre-configured scope. + +
+ +`jarvis:deny-toggle-devtools` + + + +Denies the toggle_devtools command without any pre-configured scope. + +
+ +`jarvis:allow-toggle-hidden-files` + + + +Enables the toggle_hidden_files command without any pre-configured scope. + +
+ +`jarvis:deny-toggle-hidden-files` + + + +Denies the toggle_hidden_files command without any pre-configured scope. + +
+ +`jarvis:allow-toggle-mute` + + + +Enables the toggle_mute command without any pre-configured scope. + +
+ +`jarvis:deny-toggle-mute` + + + +Denies the toggle_mute command without any pre-configured scope. + +
+ +`jarvis:allow-toggle-stage-manager` + + + +Enables the toggle_stage_manager command without any pre-configured scope. + +
+ +`jarvis:deny-toggle-stage-manager` + + + +Denies the toggle_stage_manager command without any pre-configured scope. + +
+ +`jarvis:allow-toggle-system-appearance` + + + +Enables the toggle_system_appearance command without any pre-configured scope. + +
+ +`jarvis:deny-toggle-system-appearance` + + + +Denies the toggle_system_appearance command without any pre-configured scope. + +
+ +`jarvis:allow-turn-volume-down` + + + +Enables the turn_volume_down command without any pre-configured scope. + +
+ +`jarvis:deny-turn-volume-down` + + + +Denies the turn_volume_down command without any pre-configured scope. + +
+ +`jarvis:allow-turn-volume-up` + + + +Enables the turn_volume_up command without any pre-configured scope. + +
+ +`jarvis:deny-turn-volume-up` + + + +Denies the turn_volume_up command without any pre-configured scope. + +
+ +`jarvis:allow-unmute` + + + +Enables the unmute command without any pre-configured scope. + +
+ +`jarvis:deny-unmute` + + + +Denies the unmute command without any pre-configured scope. + +
+ +`jarvis:allow-unregister-extension-spawned-process` + + + +Enables the unregister_extension_spawned_process command without any pre-configured scope. + +
+ +`jarvis:deny-unregister-extension-spawned-process` + + + +Denies the unregister_extension_spawned_process command without any pre-configured scope. + +
+ +`jarvis:allow-unregister-extension-window` + + + +Enables the unregister_extension_window command without any pre-configured scope. + +
+ +`jarvis:deny-unregister-extension-window` + + + +Denies the unregister_extension_window command without any pre-configured scope. + +
+ +`jarvis:allow-unzip` + + + +Enables the unzip command without any pre-configured scope. + +
+ +`jarvis:deny-unzip` + + + +Denies the unzip command without any pre-configured scope. + +
+ +`jarvis:allow-update-command-by-id` + + + +Enables the update_command_by_id command without any pre-configured scope. + +
+ +`jarvis:deny-update-command-by-id` + + + +Denies the update_command_by_id command without any pre-configured scope. + +
+ +`jarvis:allow-update-extension-data-by-id` + + + +Enables the update_extension_data_by_id command without any pre-configured scope. + +
+ +`jarvis:deny-update-extension-data-by-id` + + + +Denies the update_extension_data_by_id command without any pre-configured scope. + +
+ +`jarvis:allow-verify-auth` + + + +Enables the verify_auth command without any pre-configured scope. + +
+ +`jarvis:deny-verify-auth` + + + +Denies the verify_auth command without any pre-configured scope. + +
+ +`jarvis:allow-all-store` + + + +This enables all permissions for Jarvis + +
diff --git a/packages/tauri-plugins/jarvis/permissions/schemas/schema.json b/packages/tauri-plugins/jarvis/permissions/schemas/schema.json new file mode 100644 index 0000000..28823bb --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/schemas/schema.json @@ -0,0 +1,1090 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "title": "PermissionFile", + "description": "Permission file that can define a default permission, a set of permissions or a list of inlined permissions.", + "type": "object", + "properties": { + "default": { + "description": "The default permission set for the plugin", + "anyOf": [ + { + "$ref": "#/definitions/DefaultPermission" + }, + { + "type": "null" + } + ] + }, + "set": { + "description": "A list of permissions sets defined", + "type": "array", + "items": { + "$ref": "#/definitions/PermissionSet" + } + }, + "permission": { + "description": "A list of inlined permissions", + "default": [], + "type": "array", + "items": { + "$ref": "#/definitions/Permission" + } + } + }, + "definitions": { + "DefaultPermission": { + "description": "The default permission set of the plugin.\n\nWorks similarly to a permission with the \"default\" identifier.", + "type": "object", + "required": [ + "permissions" + ], + "properties": { + "version": { + "description": "The version of the permission.", + "type": [ + "integer", + "null" + ], + "format": "uint64", + "minimum": 1.0 + }, + "description": { + "description": "Human-readable description of what the permission does. Tauri convention is to use

headings in markdown content for Tauri documentation generation purposes.", + "type": [ + "string", + "null" + ] + }, + "permissions": { + "description": "All permissions this set contains.", + "type": "array", + "items": { + "type": "string" + } + } + } + }, + "PermissionSet": { + "description": "A set of direct permissions grouped together under a new name.", + "type": "object", + "required": [ + "description", + "identifier", + "permissions" + ], + "properties": { + "identifier": { + "description": "A unique identifier for the permission.", + "type": "string" + }, + "description": { + "description": "Human-readable description of what the permission does.", + "type": "string" + }, + "permissions": { + "description": "All permissions this set contains.", + "type": "array", + "items": { + "$ref": "#/definitions/PermissionKind" + } + } + } + }, + "Permission": { + "description": "Descriptions of explicit privileges of commands.\n\nIt can enable commands to be accessible in the frontend of the application.\n\nIf the scope is defined it can be used to fine grain control the access of individual or multiple commands.", + "type": "object", + "required": [ + "identifier" + ], + "properties": { + "version": { + "description": "The version of the permission.", + "type": [ + "integer", + "null" + ], + "format": "uint64", + "minimum": 1.0 + }, + "identifier": { + "description": "A unique identifier for the permission.", + "type": "string" + }, + "description": { + "description": "Human-readable description of what the permission does. Tauri internal convention is to use

headings in markdown content for Tauri documentation generation purposes.", + "type": [ + "string", + "null" + ] + }, + "commands": { + "description": "Allowed or denied commands when using this permission.", + "default": { + "allow": [], + "deny": [] + }, + "allOf": [ + { + "$ref": "#/definitions/Commands" + } + ] + }, + "scope": { + "description": "Allowed or denied scoped when using this permission.", + "allOf": [ + { + "$ref": "#/definitions/Scopes" + } + ] + }, + "platforms": { + "description": "Target platforms this permission applies. By default all platforms are affected by this permission.", + "type": [ + "array", + "null" + ], + "items": { + "$ref": "#/definitions/Target" + } + } + } + }, + "Commands": { + "description": "Allowed and denied commands inside a permission.\n\nIf two commands clash inside of `allow` and `deny`, it should be denied by default.", + "type": "object", + "properties": { + "allow": { + "description": "Allowed command.", + "default": [], + "type": "array", + "items": { + "type": "string" + } + }, + "deny": { + "description": "Denied command, which takes priority.", + "default": [], + "type": "array", + "items": { + "type": "string" + } + } + } + }, + "Scopes": { + "description": "An argument for fine grained behavior control of Tauri commands.\n\nIt can be of any serde serializable type and is used to allow or prevent certain actions inside a Tauri command. The configured scope is passed to the command and will be enforced by the command implementation.\n\n## Example\n\n```json { \"allow\": [{ \"path\": \"$HOME/**\" }], \"deny\": [{ \"path\": \"$HOME/secret.txt\" }] } ```", + "type": "object", + "properties": { + "allow": { + "description": "Data that defines what is allowed by the scope.", + "type": [ + "array", + "null" + ], + "items": { + "$ref": "#/definitions/Value" + } + }, + "deny": { + "description": "Data that defines what is denied by the scope. This should be prioritized by validation logic.", + "type": [ + "array", + "null" + ], + "items": { + "$ref": "#/definitions/Value" + } + } + } + }, + "Value": { + "description": "All supported ACL values.", + "anyOf": [ + { + "description": "Represents a null JSON value.", + "type": "null" + }, + { + "description": "Represents a [`bool`].", + "type": "boolean" + }, + { + "description": "Represents a valid ACL [`Number`].", + "allOf": [ + { + "$ref": "#/definitions/Number" + } + ] + }, + { + "description": "Represents a [`String`].", + "type": "string" + }, + { + "description": "Represents a list of other [`Value`]s.", + "type": "array", + "items": { + "$ref": "#/definitions/Value" + } + }, + { + "description": "Represents a map of [`String`] keys to [`Value`]s.", + "type": "object", + "additionalProperties": { + "$ref": "#/definitions/Value" + } + } + ] + }, + "Number": { + "description": "A valid ACL number.", + "anyOf": [ + { + "description": "Represents an [`i64`].", + "type": "integer", + "format": "int64" + }, + { + "description": "Represents a [`f64`].", + "type": "number", + "format": "double" + } + ] + }, + "Target": { + "description": "Platform target.", + "oneOf": [ + { + "description": "MacOS.", + "type": "string", + "enum": [ + "macOS" + ] + }, + { + "description": "Windows.", + "type": "string", + "enum": [ + "windows" + ] + }, + { + "description": "Linux.", + "type": "string", + "enum": [ + "linux" + ] + }, + { + "description": "Android.", + "type": "string", + "enum": [ + "android" + ] + }, + { + "description": "iOS.", + "type": "string", + "enum": [ + "iOS" + ] + } + ] + }, + "PermissionKind": { + "type": "string", + "oneOf": [ + { + "description": "This enables all permissions for Jarvis", + "type": "string", + "const": "allow-all" + }, + { + "description": "Enables the add_to_history command without any pre-configured scope.", + "type": "string", + "const": "allow-add-to-history" + }, + { + "description": "Denies the add_to_history command without any pre-configured scope.", + "type": "string", + "const": "deny-add-to-history" + }, + { + "description": "Enables the app_is_dev command without any pre-configured scope.", + "type": "string", + "const": "allow-app-is-dev" + }, + { + "description": "Denies the app_is_dev command without any pre-configured scope.", + "type": "string", + "const": "deny-app-is-dev" + }, + { + "description": "Enables the check_screen_capture_access command without any pre-configured scope.", + "type": "string", + "const": "allow-check-screen-capture-access" + }, + { + "description": "Denies the check_screen_capture_access command without any pre-configured scope.", + "type": "string", + "const": "deny-check-screen-capture-access" + }, + { + "description": "Enables the close_devtools command without any pre-configured scope.", + "type": "string", + "const": "allow-close-devtools" + }, + { + "description": "Denies the close_devtools command without any pre-configured scope.", + "type": "string", + "const": "deny-close-devtools" + }, + { + "description": "Enables the compress_tarball command without any pre-configured scope.", + "type": "string", + "const": "allow-compress-tarball" + }, + { + "description": "Denies the compress_tarball command without any pre-configured scope.", + "type": "string", + "const": "deny-compress-tarball" + }, + { + "description": "Enables the create_command command without any pre-configured scope.", + "type": "string", + "const": "allow-create-command" + }, + { + "description": "Denies the create_command command without any pre-configured scope.", + "type": "string", + "const": "deny-create-command" + }, + { + "description": "Enables the create_extension command without any pre-configured scope.", + "type": "string", + "const": "allow-create-extension" + }, + { + "description": "Denies the create_extension command without any pre-configured scope.", + "type": "string", + "const": "deny-create-extension" + }, + { + "description": "Enables the create_extension_data command without any pre-configured scope.", + "type": "string", + "const": "allow-create-extension-data" + }, + { + "description": "Denies the create_extension_data command without any pre-configured scope.", + "type": "string", + "const": "deny-create-extension-data" + }, + { + "description": "Enables the decompress_tarball command without any pre-configured scope.", + "type": "string", + "const": "allow-decompress-tarball" + }, + { + "description": "Denies the decompress_tarball command without any pre-configured scope.", + "type": "string", + "const": "deny-decompress-tarball" + }, + { + "description": "Enables the delete_command_by_id command without any pre-configured scope.", + "type": "string", + "const": "allow-delete-command-by-id" + }, + { + "description": "Denies the delete_command_by_id command without any pre-configured scope.", + "type": "string", + "const": "deny-delete-command-by-id" + }, + { + "description": "Enables the delete_extension_by_ext_id command without any pre-configured scope.", + "type": "string", + "const": "allow-delete-extension-by-ext-id" + }, + { + "description": "Denies the delete_extension_by_ext_id command without any pre-configured scope.", + "type": "string", + "const": "deny-delete-extension-by-ext-id" + }, + { + "description": "Enables the delete_extension_by_path command without any pre-configured scope.", + "type": "string", + "const": "allow-delete-extension-by-path" + }, + { + "description": "Denies the delete_extension_by_path command without any pre-configured scope.", + "type": "string", + "const": "deny-delete-extension-by-path" + }, + { + "description": "Enables the delete_extension_data_by_id command without any pre-configured scope.", + "type": "string", + "const": "allow-delete-extension-data-by-id" + }, + { + "description": "Denies the delete_extension_data_by_id command without any pre-configured scope.", + "type": "string", + "const": "deny-delete-extension-data-by-id" + }, + { + "description": "Enables the eject_all_disks command without any pre-configured scope.", + "type": "string", + "const": "allow-eject-all-disks" + }, + { + "description": "Denies the eject_all_disks command without any pre-configured scope.", + "type": "string", + "const": "deny-eject-all-disks" + }, + { + "description": "Enables the empty_trash command without any pre-configured scope.", + "type": "string", + "const": "allow-empty-trash" + }, + { + "description": "Denies the empty_trash command without any pre-configured scope.", + "type": "string", + "const": "deny-empty-trash" + }, + { + "description": "Enables the file_search command without any pre-configured scope.", + "type": "string", + "const": "allow-file-search" + }, + { + "description": "Denies the file_search command without any pre-configured scope.", + "type": "string", + "const": "deny-file-search" + }, + { + "description": "Enables the get_all_extensions command without any pre-configured scope.", + "type": "string", + "const": "allow-get-all-extensions" + }, + { + "description": "Denies the get_all_extensions command without any pre-configured scope.", + "type": "string", + "const": "deny-get-all-extensions" + }, + { + "description": "Enables the get_all_extensions_by_identifier command without any pre-configured scope.", + "type": "string", + "const": "allow-get-all-extensions-by-identifier" + }, + { + "description": "Denies the get_all_extensions_by_identifier command without any pre-configured scope.", + "type": "string", + "const": "deny-get-all-extensions-by-identifier" + }, + { + "description": "Enables the get_applications command without any pre-configured scope.", + "type": "string", + "const": "allow-get-applications" + }, + { + "description": "Denies the get_applications command without any pre-configured scope.", + "type": "string", + "const": "deny-get-applications" + }, + { + "description": "Enables the get_command_by_id command without any pre-configured scope.", + "type": "string", + "const": "allow-get-command-by-id" + }, + { + "description": "Denies the get_command_by_id command without any pre-configured scope.", + "type": "string", + "const": "deny-get-command-by-id" + }, + { + "description": "Enables the get_commands_by_ext_id command without any pre-configured scope.", + "type": "string", + "const": "allow-get-commands-by-ext-id" + }, + { + "description": "Denies the get_commands_by_ext_id command without any pre-configured scope.", + "type": "string", + "const": "deny-get-commands-by-ext-id" + }, + { + "description": "Enables the get_default_extensions_dir command without any pre-configured scope.", + "type": "string", + "const": "allow-get-default-extensions-dir" + }, + { + "description": "Denies the get_default_extensions_dir command without any pre-configured scope.", + "type": "string", + "const": "deny-get-default-extensions-dir" + }, + { + "description": "Enables the get_default_extensions_storage_dir command without any pre-configured scope.", + "type": "string", + "const": "allow-get-default-extensions-storage-dir" + }, + { + "description": "Denies the get_default_extensions_storage_dir command without any pre-configured scope.", + "type": "string", + "const": "deny-get-default-extensions-storage-dir" + }, + { + "description": "Enables the get_dev_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "allow-get-dev-extension-folder" + }, + { + "description": "Denies the get_dev_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "deny-get-dev-extension-folder" + }, + { + "description": "Enables the get_ext_label_map command without any pre-configured scope.", + "type": "string", + "const": "allow-get-ext-label-map" + }, + { + "description": "Denies the get_ext_label_map command without any pre-configured scope.", + "type": "string", + "const": "deny-get-ext-label-map" + }, + { + "description": "Enables the get_extension_data_by_id command without any pre-configured scope.", + "type": "string", + "const": "allow-get-extension-data-by-id" + }, + { + "description": "Denies the get_extension_data_by_id command without any pre-configured scope.", + "type": "string", + "const": "deny-get-extension-data-by-id" + }, + { + "description": "Enables the get_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "allow-get-extension-folder" + }, + { + "description": "Denies the get_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "deny-get-extension-folder" + }, + { + "description": "Enables the get_history command without any pre-configured scope.", + "type": "string", + "const": "allow-get-history" + }, + { + "description": "Denies the get_history command without any pre-configured scope.", + "type": "string", + "const": "deny-get-history" + }, + { + "description": "Enables the get_peers command without any pre-configured scope.", + "type": "string", + "const": "allow-get-peers" + }, + { + "description": "Denies the get_peers command without any pre-configured scope.", + "type": "string", + "const": "deny-get-peers" + }, + { + "description": "Enables the get_selected_files_in_file_explorer command without any pre-configured scope.", + "type": "string", + "const": "allow-get-selected-files-in-file-explorer" + }, + { + "description": "Denies the get_selected_files_in_file_explorer command without any pre-configured scope.", + "type": "string", + "const": "deny-get-selected-files-in-file-explorer" + }, + { + "description": "Enables the get_server_port command without any pre-configured scope.", + "type": "string", + "const": "allow-get-server-port" + }, + { + "description": "Denies the get_server_port command without any pre-configured scope.", + "type": "string", + "const": "deny-get-server-port" + }, + { + "description": "Enables the get_unique_extension_by_identifier command without any pre-configured scope.", + "type": "string", + "const": "allow-get-unique-extension-by-identifier" + }, + { + "description": "Denies the get_unique_extension_by_identifier command without any pre-configured scope.", + "type": "string", + "const": "deny-get-unique-extension-by-identifier" + }, + { + "description": "Enables the get_unique_extension_by_path command without any pre-configured scope.", + "type": "string", + "const": "allow-get-unique-extension-by-path" + }, + { + "description": "Denies the get_unique_extension_by_path command without any pre-configured scope.", + "type": "string", + "const": "deny-get-unique-extension-by-path" + }, + { + "description": "Enables the hide_all_apps_except_frontmost command without any pre-configured scope.", + "type": "string", + "const": "allow-hide-all-apps-except-frontmost" + }, + { + "description": "Denies the hide_all_apps_except_frontmost command without any pre-configured scope.", + "type": "string", + "const": "deny-hide-all-apps-except-frontmost" + }, + { + "description": "Enables the is_devtools_open command without any pre-configured scope.", + "type": "string", + "const": "allow-is-devtools-open" + }, + { + "description": "Denies the is_devtools_open command without any pre-configured scope.", + "type": "string", + "const": "deny-is-devtools-open" + }, + { + "description": "Enables the is_window_label_registered command without any pre-configured scope.", + "type": "string", + "const": "allow-is-window-label-registered" + }, + { + "description": "Denies the is_window_label_registered command without any pre-configured scope.", + "type": "string", + "const": "deny-is-window-label-registered" + }, + { + "description": "Enables the logout_user command without any pre-configured scope.", + "type": "string", + "const": "allow-logout-user" + }, + { + "description": "Denies the logout_user command without any pre-configured scope.", + "type": "string", + "const": "deny-logout-user" + }, + { + "description": "Enables the mute command without any pre-configured scope.", + "type": "string", + "const": "allow-mute" + }, + { + "description": "Denies the mute command without any pre-configured scope.", + "type": "string", + "const": "deny-mute" + }, + { + "description": "Enables the open_devtools command without any pre-configured scope.", + "type": "string", + "const": "allow-open-devtools" + }, + { + "description": "Denies the open_devtools command without any pre-configured scope.", + "type": "string", + "const": "deny-open-devtools" + }, + { + "description": "Enables the open_trash command without any pre-configured scope.", + "type": "string", + "const": "allow-open-trash" + }, + { + "description": "Denies the open_trash command without any pre-configured scope.", + "type": "string", + "const": "deny-open-trash" + }, + { + "description": "Enables the path_exists command without any pre-configured scope.", + "type": "string", + "const": "allow-path-exists" + }, + { + "description": "Denies the path_exists command without any pre-configured scope.", + "type": "string", + "const": "deny-path-exists" + }, + { + "description": "Enables the plist_to_json command without any pre-configured scope.", + "type": "string", + "const": "allow-plist-to-json" + }, + { + "description": "Denies the plist_to_json command without any pre-configured scope.", + "type": "string", + "const": "deny-plist-to-json" + }, + { + "description": "Enables the quit_all_apps command without any pre-configured scope.", + "type": "string", + "const": "allow-quit-all-apps" + }, + { + "description": "Denies the quit_all_apps command without any pre-configured scope.", + "type": "string", + "const": "deny-quit-all-apps" + }, + { + "description": "Enables the reboot command without any pre-configured scope.", + "type": "string", + "const": "allow-reboot" + }, + { + "description": "Denies the reboot command without any pre-configured scope.", + "type": "string", + "const": "deny-reboot" + }, + { + "description": "Enables the refresh_applications_list command without any pre-configured scope.", + "type": "string", + "const": "allow-refresh-applications-list" + }, + { + "description": "Denies the refresh_applications_list command without any pre-configured scope.", + "type": "string", + "const": "deny-refresh-applications-list" + }, + { + "description": "Enables the refresh_applications_list_in_bg command without any pre-configured scope.", + "type": "string", + "const": "allow-refresh-applications-list-in-bg" + }, + { + "description": "Denies the refresh_applications_list_in_bg command without any pre-configured scope.", + "type": "string", + "const": "deny-refresh-applications-list-in-bg" + }, + { + "description": "Enables the register_extension_spawned_process command without any pre-configured scope.", + "type": "string", + "const": "allow-register-extension-spawned-process" + }, + { + "description": "Denies the register_extension_spawned_process command without any pre-configured scope.", + "type": "string", + "const": "deny-register-extension-spawned-process" + }, + { + "description": "Enables the register_extension_window command without any pre-configured scope.", + "type": "string", + "const": "allow-register-extension-window" + }, + { + "description": "Denies the register_extension_window command without any pre-configured scope.", + "type": "string", + "const": "deny-register-extension-window" + }, + { + "description": "Enables the request_screen_capture_access command without any pre-configured scope.", + "type": "string", + "const": "allow-request-screen-capture-access" + }, + { + "description": "Denies the request_screen_capture_access command without any pre-configured scope.", + "type": "string", + "const": "deny-request-screen-capture-access" + }, + { + "description": "Enables the restart_server command without any pre-configured scope.", + "type": "string", + "const": "allow-restart-server" + }, + { + "description": "Denies the restart_server command without any pre-configured scope.", + "type": "string", + "const": "deny-restart-server" + }, + { + "description": "Enables the run_apple_script command without any pre-configured scope.", + "type": "string", + "const": "allow-run-apple-script" + }, + { + "description": "Denies the run_apple_script command without any pre-configured scope.", + "type": "string", + "const": "deny-run-apple-script" + }, + { + "description": "Enables the run_powershell command without any pre-configured scope.", + "type": "string", + "const": "allow-run-powershell" + }, + { + "description": "Denies the run_powershell command without any pre-configured scope.", + "type": "string", + "const": "deny-run-powershell" + }, + { + "description": "Enables the search_extension_data command without any pre-configured scope.", + "type": "string", + "const": "allow-search-extension-data" + }, + { + "description": "Denies the search_extension_data command without any pre-configured scope.", + "type": "string", + "const": "deny-search-extension-data" + }, + { + "description": "Enables the server_is_running command without any pre-configured scope.", + "type": "string", + "const": "allow-server-is-running" + }, + { + "description": "Denies the server_is_running command without any pre-configured scope.", + "type": "string", + "const": "deny-server-is-running" + }, + { + "description": "Enables the set_dev_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "allow-set-dev-extension-folder" + }, + { + "description": "Denies the set_dev_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "deny-set-dev-extension-folder" + }, + { + "description": "Enables the set_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "allow-set-extension-folder" + }, + { + "description": "Denies the set_extension_folder command without any pre-configured scope.", + "type": "string", + "const": "deny-set-extension-folder" + }, + { + "description": "Enables the set_volume command without any pre-configured scope.", + "type": "string", + "const": "allow-set-volume" + }, + { + "description": "Denies the set_volume command without any pre-configured scope.", + "type": "string", + "const": "deny-set-volume" + }, + { + "description": "Enables the show_desktop command without any pre-configured scope.", + "type": "string", + "const": "allow-show-desktop" + }, + { + "description": "Denies the show_desktop command without any pre-configured scope.", + "type": "string", + "const": "deny-show-desktop" + }, + { + "description": "Enables the shutdown command without any pre-configured scope.", + "type": "string", + "const": "allow-shutdown" + }, + { + "description": "Denies the shutdown command without any pre-configured scope.", + "type": "string", + "const": "deny-shutdown" + }, + { + "description": "Enables the sleep command without any pre-configured scope.", + "type": "string", + "const": "allow-sleep" + }, + { + "description": "Denies the sleep command without any pre-configured scope.", + "type": "string", + "const": "deny-sleep" + }, + { + "description": "Enables the sleep_displays command without any pre-configured scope.", + "type": "string", + "const": "allow-sleep-displays" + }, + { + "description": "Denies the sleep_displays command without any pre-configured scope.", + "type": "string", + "const": "deny-sleep-displays" + }, + { + "description": "Enables the start_server command without any pre-configured scope.", + "type": "string", + "const": "allow-start-server" + }, + { + "description": "Denies the start_server command without any pre-configured scope.", + "type": "string", + "const": "deny-start-server" + }, + { + "description": "Enables the stop_server command without any pre-configured scope.", + "type": "string", + "const": "allow-stop-server" + }, + { + "description": "Denies the stop_server command without any pre-configured scope.", + "type": "string", + "const": "deny-stop-server" + }, + { + "description": "Enables the toggle_bluetooth command without any pre-configured scope.", + "type": "string", + "const": "allow-toggle-bluetooth" + }, + { + "description": "Denies the toggle_bluetooth command without any pre-configured scope.", + "type": "string", + "const": "deny-toggle-bluetooth" + }, + { + "description": "Enables the toggle_devtools command without any pre-configured scope.", + "type": "string", + "const": "allow-toggle-devtools" + }, + { + "description": "Denies the toggle_devtools command without any pre-configured scope.", + "type": "string", + "const": "deny-toggle-devtools" + }, + { + "description": "Enables the toggle_hidden_files command without any pre-configured scope.", + "type": "string", + "const": "allow-toggle-hidden-files" + }, + { + "description": "Denies the toggle_hidden_files command without any pre-configured scope.", + "type": "string", + "const": "deny-toggle-hidden-files" + }, + { + "description": "Enables the toggle_mute command without any pre-configured scope.", + "type": "string", + "const": "allow-toggle-mute" + }, + { + "description": "Denies the toggle_mute command without any pre-configured scope.", + "type": "string", + "const": "deny-toggle-mute" + }, + { + "description": "Enables the toggle_stage_manager command without any pre-configured scope.", + "type": "string", + "const": "allow-toggle-stage-manager" + }, + { + "description": "Denies the toggle_stage_manager command without any pre-configured scope.", + "type": "string", + "const": "deny-toggle-stage-manager" + }, + { + "description": "Enables the toggle_system_appearance command without any pre-configured scope.", + "type": "string", + "const": "allow-toggle-system-appearance" + }, + { + "description": "Denies the toggle_system_appearance command without any pre-configured scope.", + "type": "string", + "const": "deny-toggle-system-appearance" + }, + { + "description": "Enables the turn_volume_down command without any pre-configured scope.", + "type": "string", + "const": "allow-turn-volume-down" + }, + { + "description": "Denies the turn_volume_down command without any pre-configured scope.", + "type": "string", + "const": "deny-turn-volume-down" + }, + { + "description": "Enables the turn_volume_up command without any pre-configured scope.", + "type": "string", + "const": "allow-turn-volume-up" + }, + { + "description": "Denies the turn_volume_up command without any pre-configured scope.", + "type": "string", + "const": "deny-turn-volume-up" + }, + { + "description": "Enables the unmute command without any pre-configured scope.", + "type": "string", + "const": "allow-unmute" + }, + { + "description": "Denies the unmute command without any pre-configured scope.", + "type": "string", + "const": "deny-unmute" + }, + { + "description": "Enables the unregister_extension_spawned_process command without any pre-configured scope.", + "type": "string", + "const": "allow-unregister-extension-spawned-process" + }, + { + "description": "Denies the unregister_extension_spawned_process command without any pre-configured scope.", + "type": "string", + "const": "deny-unregister-extension-spawned-process" + }, + { + "description": "Enables the unregister_extension_window command without any pre-configured scope.", + "type": "string", + "const": "allow-unregister-extension-window" + }, + { + "description": "Denies the unregister_extension_window command without any pre-configured scope.", + "type": "string", + "const": "deny-unregister-extension-window" + }, + { + "description": "Enables the unzip command without any pre-configured scope.", + "type": "string", + "const": "allow-unzip" + }, + { + "description": "Denies the unzip command without any pre-configured scope.", + "type": "string", + "const": "deny-unzip" + }, + { + "description": "Enables the update_command_by_id command without any pre-configured scope.", + "type": "string", + "const": "allow-update-command-by-id" + }, + { + "description": "Denies the update_command_by_id command without any pre-configured scope.", + "type": "string", + "const": "deny-update-command-by-id" + }, + { + "description": "Enables the update_extension_data_by_id command without any pre-configured scope.", + "type": "string", + "const": "allow-update-extension-data-by-id" + }, + { + "description": "Denies the update_extension_data_by_id command without any pre-configured scope.", + "type": "string", + "const": "deny-update-extension-data-by-id" + }, + { + "description": "Enables the verify_auth command without any pre-configured scope.", + "type": "string", + "const": "allow-verify-auth" + }, + { + "description": "Denies the verify_auth command without any pre-configured scope.", + "type": "string", + "const": "deny-verify-auth" + }, + { + "description": "This enables all permissions for Jarvis", + "type": "string", + "const": "allow-all-store" + } + ] + } + } +} \ No newline at end of file diff --git a/packages/tauri-plugins/jarvis/permissions/store-all.toml b/packages/tauri-plugins/jarvis/permissions/store-all.toml new file mode 100644 index 0000000..2d0adb0 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/store-all.toml @@ -0,0 +1,19 @@ +"$schema" = "schemas/schema.json" + +[[permission]] +identifier = "allow-all-store" +description = "This enables all permissions for Jarvis" +commands.allow = [ + "ext_store_wrapper_set", + "ext_store_wrapper_get", + "ext_store_wrapper_has", + "ext_store_wrapper_delete", + "ext_store_wrapper_clear", + "ext_store_wrapper_reset", + "ext_store_wrapper_keys", + "ext_store_wrapper_values", + "ext_store_wrapper_entries", + "ext_store_wrapper_length", + "ext_store_wrapper_load", + "ext_store_wrapper_save", +] \ No newline at end of file diff --git a/packages/tauri-plugins/jarvis/proto/helloworld.proto b/packages/tauri-plugins/jarvis/proto/helloworld.proto new file mode 100644 index 0000000..7a4a0ee --- /dev/null +++ b/packages/tauri-plugins/jarvis/proto/helloworld.proto @@ -0,0 +1,14 @@ +syntax = "proto3"; +package helloworld; + +service Greeter { + rpc SayHello (HelloRequest) returns (HelloReply); +} + +message HelloRequest { + string name = 1; +} + +message HelloReply { + string message = 1; +} \ No newline at end of file diff --git a/packages/tauri-plugins/jarvis/self_signed_certs/cert.pem b/packages/tauri-plugins/jarvis/self_signed_certs/cert.pem new file mode 100644 index 0000000..c3e6ed2 --- /dev/null +++ b/packages/tauri-plugins/jarvis/self_signed_certs/cert.pem @@ -0,0 +1,10 @@ +-----BEGIN CERTIFICATE----- +MIIBUTCB+aADAgECAgkA9wj7pE/H2nUwCgYIKoZIzj0EAwIwITEfMB0GA1UEAwwW +cmNnZW4gc2VsZiBzaWduZWQgY2VydDAgFw03NTAxMDEwMDAwMDBaGA80MDk2MDEw +MTAwMDAwMFowITEfMB0GA1UEAwwWcmNnZW4gc2VsZiBzaWduZWQgY2VydDBZMBMG +ByqGSM49AgEGCCqGSM49AwEHA0IABE2MBAa4jgIYuqWLcQ1VKGYs8xAzYok9H6Yp +2tWRcC3ZKdm8QFlEmMJC/sY0d+Z49L9b977PUxGcE5Zsym6e4oajGDAWMBQGA1Ud +EQQNMAuCCWxvY2FsaG9zdDAKBggqhkjOPQQDAgNHADBEAiBbqzKRw01Oeq91znQY +KmCTBn/nDHSyGn5CMVsvPlOKJgIgNvf2wrefmuOzzrJx22RWV2KIvc2zt7HNBso9 +fJKS3PM= +-----END CERTIFICATE----- diff --git a/packages/tauri-plugins/jarvis/self_signed_certs/key.pem b/packages/tauri-plugins/jarvis/self_signed_certs/key.pem new file mode 100644 index 0000000..acea631 --- /dev/null +++ b/packages/tauri-plugins/jarvis/self_signed_certs/key.pem @@ -0,0 +1,5 @@ +-----BEGIN PRIVATE KEY----- +MIGHAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBG0wawIBAQQgHIWwXFtJccIhUD4J +r6M3DrW3n4FBa6agiLXo2XKczt2hRANCAARNjAQGuI4CGLqli3ENVShmLPMQM2KJ +PR+mKdrVkXAt2SnZvEBZRJjCQv7GNHfmePS/W/e+z1MRnBOWbMpunuKG +-----END PRIVATE KEY----- diff --git a/packages/tauri-plugins/jarvis/src/commands/apps.rs b/packages/tauri-plugins/jarvis/src/commands/apps.rs new file mode 100644 index 0000000..c459c3d --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/apps.rs @@ -0,0 +1,34 @@ +use applications::{App, AppInfo, AppInfoContext}; +use std::sync::Mutex; + +#[derive(Default)] +pub struct ApplicationsState { + ctx: Mutex, +} +#[tauri::command] +pub async fn get_applications( + state: tauri::State<'_, ApplicationsState>, +) -> Result, String> { + Ok(state.ctx.lock().unwrap().get_all_apps()) +} + +#[tauri::command] +pub async fn refresh_applications_list( + state: tauri::State<'_, ApplicationsState>, +) -> Result<(), String> { + state + .ctx + .lock() + .unwrap() + .refresh_apps() + .map_err(|e| e.to_string())?; + Ok(()) +} + +#[tauri::command] +pub async fn refresh_applications_list_in_bg( + state: tauri::State<'_, ApplicationsState>, +) -> Result<(), String> { + state.ctx.lock().unwrap().refresh_apps_in_background(); + Ok(()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/clipboard.rs b/packages/tauri-plugins/jarvis/src/commands/clipboard.rs new file mode 100644 index 0000000..a844179 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/clipboard.rs @@ -0,0 +1,36 @@ +use super::db::DBState; +use crate::model::clipboard_history::{ClipboardContentType, ClipboardHistory, Record}; +use std::time::SystemTime; + +#[tauri::command] +pub async fn get_history( + state: tauri::State<'_, ClipboardHistory>, + db: tauri::State<'_, DBState>, +) -> Result, String> { + state.get_all_records().map_err(|err| err.to_string()) + // let mut history = state.history.lock().unwrap().to_vec(); + // history.reverse(); + // // only return the first 100 items + // history.truncate(100); + // Ok(history) +} + +#[tauri::command] +pub async fn add_to_history( + state: tauri::State<'_, ClipboardHistory>, + value: String, + text: String, + content_type: ClipboardContentType, +) -> Result<(), String> { + state + .add_record(Record { + timestamp: SystemTime::now() + .duration_since(SystemTime::UNIX_EPOCH) + .unwrap() + .as_millis(), + content_type, + value, + text, + }) + .map_err(|err| err.to_string()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/db.rs b/packages/tauri-plugins/jarvis/src/commands/db.rs new file mode 100644 index 0000000..cc3aa0d --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/db.rs @@ -0,0 +1,266 @@ +use db::{ + models::{Cmd, CmdType, Ext, ExtData}, + ExtDataField, JarvisDB, SQLSortOrder, +}; +use std::{path::PathBuf, sync::Mutex}; +use tauri::{utils::acl::identifier, State}; + +#[derive(Debug)] +pub struct DBState { + pub db: Mutex, + // pub peers: Mutex>, +} + +impl DBState { + pub fn new(path: PathBuf, key: Option) -> anyhow::Result { + let db = JarvisDB::new(path, key)?; + db.init()?; + Ok(Self { db: Mutex::new(db) }) + } +} + +/* -------------------------------------------------------------------------- */ +/* Extension CRUD */ +/* -------------------------------------------------------------------------- */ +#[tauri::command] +pub async fn create_extension( + db: State<'_, DBState>, + identifier: &str, + version: &str, + enabled: Option, + path: Option<&str>, + data: Option<&str>, +) -> Result<(), String> { + db.db + .lock() + .unwrap() + .create_extension(identifier, version, enabled.unwrap_or(true), path, data) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn get_all_extensions(db: State<'_, DBState>) -> Result, String> { + db.db + .lock() + .unwrap() + .get_all_extensions() + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn get_all_extensions_by_identifier( + identifier: &str, + db: State<'_, DBState>, +) -> Result, String> { + db.db + .lock() + .unwrap() + .get_all_extensions_by_identifier(identifier) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn get_unique_extension_by_identifier( + identifier: &str, + db: State<'_, DBState>, +) -> Result, String> { + let ext = db + .db + .lock() + .unwrap() + .get_unique_extension_by_identifier(identifier) + .map_err(|err| err.to_string())?; + Ok(ext) +} + +#[tauri::command] +pub async fn get_unique_extension_by_path( + path: &str, + db: State<'_, DBState>, +) -> Result, String> { + let ext = db + .db + .lock() + .unwrap() + .get_unique_extension_by_path(path) + .map_err(|err| err.to_string())?; + Ok(ext) +} + +#[tauri::command] +pub async fn delete_extension_by_path(path: &str, db: State<'_, DBState>) -> Result<(), String> { + db.db + .lock() + .unwrap() + .delete_extension_by_path(path) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn delete_extension_by_ext_id(ext_id: i32, db: State<'_, DBState>) -> Result<(), String> { + db.db + .lock() + .unwrap() + .delete_extension_by_ext_id(ext_id) + .map_err(|err| err.to_string()) +} + +/* -------------------------------------------------------------------------- */ +/* Extension Command CRUD */ +/* -------------------------------------------------------------------------- */ + +#[tauri::command] +pub async fn create_command( + db: State<'_, DBState>, + ext_id: i32, + name: &str, + cmd_type: CmdType, + data: &str, + enabled: bool, + alias: Option<&str>, + hotkey: Option<&str>, +) -> Result<(), String> { + db.db + .lock() + .unwrap() + .create_command(ext_id, name, cmd_type, data, enabled, alias, hotkey) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn get_command_by_id(db: State<'_, DBState>, cmd_id: i32) -> Result, String> { + db.db + .lock() + .unwrap() + .get_command_by_id(cmd_id) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn get_commands_by_ext_id( + db: State<'_, DBState>, + ext_id: i32, +) -> Result, String> { + db.db + .lock() + .unwrap() + .get_commands_by_ext_id(ext_id) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn delete_command_by_id(db: State<'_, DBState>, cmd_id: i32) -> Result<(), String> { + db.db + .lock() + .unwrap() + .delete_command_by_id(cmd_id) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn update_command_by_id( + db: State<'_, DBState>, + cmd_id: i32, + name: &str, + cmd_type: CmdType, + data: &str, + enabled: bool, + alias: Option<&str>, + hotkey: Option<&str>, +) -> Result<(), String> { + db.db + .lock() + .unwrap() + .update_command_by_id(cmd_id, name, cmd_type, data, enabled, alias, hotkey) + .map_err(|err| err.to_string()) +} + +/* -------------------------------------------------------------------------- */ +/* Extension Data CRUD */ +/* -------------------------------------------------------------------------- */ +#[tauri::command] +pub async fn create_extension_data( + ext_id: i32, + data_type: &str, + data: &str, + search_text: Option<&str>, + db: State<'_, DBState>, +) -> Result<(), String> { + db.db + .lock() + .unwrap() + .create_extension_data(ext_id, data_type, data, search_text) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn get_extension_data_by_id( + data_id: i32, + db: State<'_, DBState>, +) -> Result, String> { + db.db + .lock() + .unwrap() + .get_extension_data_by_id(data_id) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn search_extension_data( + ext_id: i32, + search_exact_match: bool, + data_id: Option, + data_type: Option<&str>, + search_text: Option<&str>, + after_created_at: Option<&str>, + before_created_at: Option<&str>, + db: State<'_, DBState>, + limit: Option, + order_by_created_at: Option, + order_by_updated_at: Option, + fields: Option>, +) -> Result, String> { + db.db + .lock() + .unwrap() + .search_extension_data( + ext_id, + search_exact_match, + data_id, + data_type, + search_text, + after_created_at, + before_created_at, + limit, + order_by_created_at, + order_by_updated_at, + fields, + ) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn delete_extension_data_by_id( + data_id: i32, + db: State<'_, DBState>, +) -> Result<(), String> { + db.db + .lock() + .unwrap() + .delete_extension_data_by_id(data_id) + .map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn update_extension_data_by_id( + data_id: i32, + data: &str, + search_text: Option<&str>, + db: State<'_, DBState>, +) -> Result<(), String> { + db.db + .lock() + .unwrap() + .update_extension_data_by_id(data_id, data, search_text) + .map_err(|err| err.to_string()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/dev.rs b/packages/tauri-plugins/jarvis/src/commands/dev.rs new file mode 100644 index 0000000..fbdb80d --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/dev.rs @@ -0,0 +1,60 @@ +use tauri::{Manager, Runtime}; + +#[tauri::command] +pub fn open_devtools(window: tauri::Window) -> Result<(), String> { + #[cfg(debug_assertions)] + { + let win = window.get_webview_window(window.label()); + if let Some(win) = win { + win.open_devtools(); + } + } + Ok(()) +} + +#[tauri::command] +pub fn close_devtools(window: tauri::Window) -> Result<(), String> { + #[cfg(debug_assertions)] + { + let win = window.get_webview_window(window.label()); + if let Some(win) = win { + win.close_devtools(); + } + } + Ok(()) +} + +#[tauri::command] +pub fn is_devtools_open(window: tauri::Window) -> Result { + #[cfg(debug_assertions)] + { + let win = window.get_webview_window(window.label()); + if let Some(win) = win { + Ok(win.is_devtools_open()) + } else { + Err("Window not found".to_string()) + } + } + #[cfg(not(debug_assertions))] + Err("Devtools is not available in release mode".to_string()) +} + +#[tauri::command] +pub fn toggle_devtools(window: tauri::Window) -> Result<(), String> { + #[cfg(debug_assertions)] + { + let is_open = + is_devtools_open(window.clone()).expect("failed to check if devtools is open"); + if is_open { + close_devtools(window).expect("failed to close devtools"); + } else { + open_devtools(window).expect("failed to open devtools"); + } + } + Ok(()) +} + +#[tauri::command] +pub fn app_is_dev() -> Result { + Ok(tauri::is_dev()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/discovery.rs b/packages/tauri-plugins/jarvis/src/commands/discovery.rs new file mode 100644 index 0000000..640d511 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/discovery.rs @@ -0,0 +1,43 @@ +use std::{collections::HashMap, sync::Mutex}; +use tauri_plugin_network::network::mdns::ServiceInfoMod; + +#[derive(Default, Debug)] +pub struct Peers { + pub peers: Mutex>, +} + +impl Peers { + pub fn add_peer(&self, peer: ServiceInfoMod) { + let mut peers = self.peers.lock().unwrap(); + peers.insert(peer.hostname.clone(), peer); + } + + pub fn remove_peer(&self, service_type: String, fullname: String) { + let peers = self.peers.lock().unwrap(); + // find the peer by service_type and fullname + let peer = peers + .iter() + .find(|(_, peer)| peer.fullname == fullname && peer.service_type == service_type); + if let Some((hostname, _)) = peer { + self.peers.lock().unwrap().remove(hostname); + } + } + + pub fn clear(&self) { + let mut peers = self.peers.lock().unwrap(); + peers.clear(); + } + + pub fn set_peers(&self, peers: HashMap) { + self.clear(); + self.peers.lock().unwrap().extend(peers); + } +} + +#[tauri::command] +pub async fn get_peers( + state: tauri::State<'_, Peers>, +) -> Result, String> { + let _peers = state.peers.lock().unwrap(); + Ok(_peers.to_owned()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/extension.rs b/packages/tauri-plugins/jarvis/src/commands/extension.rs new file mode 100644 index 0000000..10af74b --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/extension.rs @@ -0,0 +1,150 @@ +use crate::JarvisState; +use crate::{ + model::{ + extension::Extension, + manifest::{ExtPackageJsonExtra, MANIFEST_FILE_NAME}, + }, + utils::manifest::load_jarvis_ext_manifest, +}; +use std::collections::HashMap; +use std::{fmt::format, path::PathBuf}; +use tauri::{command, AppHandle, Runtime, State, Window}; + +/// manifest_path can be folder of package.json +/// If it's a folder, join it with package.json +#[tauri::command] +pub async fn load_manifest( + _app: tauri::AppHandle, + _window: tauri::Window, + manifest_path: PathBuf, +) -> Result { + Ok(ExtPackageJsonExtra::from( + load_jarvis_ext_manifest(manifest_path.clone()).map_err(|e| e.to_string())?, + manifest_path.parent().unwrap().to_path_buf(), + )) +} + +#[tauri::command] +pub async fn load_all_extensions( + _app: tauri::AppHandle, + _window: tauri::Window, + extensions_folder: PathBuf, +) -> Result, String> { + let mut extensions_with_path: Vec = vec![]; + for entry in std::fs::read_dir(extensions_folder).map_err(|e| e.to_string())? { + let entry = entry.map_err(|e| e.to_string())?; + + if entry.path().join(MANIFEST_FILE_NAME).exists() { + let ext_manifest = load_jarvis_ext_manifest(entry.path()).map_err(|e| e.to_string()); + if ext_manifest.is_err() { + continue; + } + extensions_with_path.push(ExtPackageJsonExtra::from( + ext_manifest.unwrap(), + entry.path(), + )); + } + } + Ok(extensions_with_path) +} + +#[tauri::command] +pub async fn is_window_label_registered( + _app: AppHandle, + state: State<'_, JarvisState>, + label: String, +) -> Result { + Ok(state + .window_label_ext_map + .lock() + .unwrap() + .contains_key(label.as_str())) +} + +#[tauri::command] +pub async fn register_extension_window( + _app: AppHandle, + state: State<'_, JarvisState>, + extension_path: PathBuf, + dist: Option, + window_label: Option, +) -> Result { + let window_label_2 = match window_label { + Some(label) => label, + None => format!("main:ext:{}", uuid::Uuid::new_v4()), + }; + let mut label_ext_map = state.window_label_ext_map.lock().unwrap(); + // if label_ext_map.contains_key(window_label_2.as_str()) { + // return Err(format!( + // "Window with label {} is already registered", + // &window_label_2 + // )); + // } + let ext = Extension { + path: extension_path, + processes: vec![], + dist: dist, + // identifier: manifest.kunkun.identifier, + }; + label_ext_map.insert(window_label_2.clone(), ext); + Ok(window_label_2) +} + +#[tauri::command] +pub async fn register_extension_spawned_process( + _app: AppHandle, + state: State<'_, JarvisState>, + window_label: String, + pid: u32, +) -> Result<(), String> { + let mut label_ext_map = state.window_label_ext_map.lock().unwrap(); + // check if window_label is registered, if not, return error + if !label_ext_map.contains_key(window_label.as_str()) { + return Err(format!( + "Window with label {} is not registered", + &window_label + )); + } + let ext = label_ext_map.get_mut(window_label.as_str()).unwrap(); + ext.processes.push(pid); + Ok(()) +} + +#[tauri::command] +pub async fn unregister_extension_spawned_process( + _app: AppHandle, + state: State<'_, JarvisState>, + window_label: String, + pid: u32, +) -> Result<(), String> { + let mut label_ext_map = state.window_label_ext_map.lock().unwrap(); + label_ext_map + .get_mut(window_label.as_str()) + .unwrap() + .processes + .retain(|p| *p != pid); + Ok(()) +} + +#[tauri::command] +pub async fn get_ext_label_map( + _app: AppHandle, + _window: Window, + state: State<'_, JarvisState>, +) -> Result, String> { + Ok(state.window_label_ext_map.lock().unwrap().clone()) +} + +#[tauri::command] +pub async fn unregister_extension_window( + _app: AppHandle, + state: State<'_, JarvisState>, + label: String, +) -> Result { + Ok(state + .window_label_ext_map + .lock() + .unwrap() + .remove(label.as_str()) + .is_some()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/file_search.rs b/packages/tauri-plugins/jarvis/src/commands/file_search.rs new file mode 100644 index 0000000..663dfd1 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/file_search.rs @@ -0,0 +1,85 @@ +use rust_search::{FileSize, FilterExt, SearchBuilder}; +use std::{ + path::PathBuf, + time::{Duration, SystemTime}, +}; + +#[derive(serde::Deserialize, serde::Serialize)] +pub struct SearchParams { + locations: Vec, + query: Option, + ext: Option, + depth: Option, + limit: Option, + hidden: bool, + ignore_case: bool, + file_size_greater: Option, + file_size_smaller: Option, + file_size_equal: Option, + created_after: Option, + created_before: Option, + modified_after: Option, + modified_before: Option, +} + +fn search(search_params: SearchParams) -> Vec { + let mut searcher = SearchBuilder::default(); + if search_params.locations.len() == 1 { + searcher = searcher.location(search_params.locations[0].clone()); + } else { + searcher = searcher.more_locations(search_params.locations); + } + if let Some(query) = search_params.query { + searcher = searcher.search_input(query); + } + if let Some(ext) = search_params.ext { + searcher = searcher.ext(ext); + } + if let Some(depth) = search_params.depth { + searcher = searcher.depth(depth); + } + if let Some(limit) = search_params.limit { + searcher = searcher.limit(limit); + } + if let Some(file_size_smaller) = search_params.file_size_smaller { + searcher = searcher.file_size_smaller(FileSize::Byte(file_size_smaller)); + } + if let Some(file_size_greater) = search_params.file_size_greater { + searcher = searcher.file_size_greater(FileSize::Byte(file_size_greater)); + } + if let Some(file_size_equal) = search_params.file_size_equal { + searcher = searcher.file_size_equal(FileSize::Byte(file_size_equal)); + } + if let Some(created_after) = search_params.created_after { + searcher = + searcher.created_after(SystemTime::UNIX_EPOCH + Duration::from_secs(created_after)); + } + if let Some(created_before) = search_params.created_before { + searcher = + searcher.created_before(SystemTime::UNIX_EPOCH + Duration::from_secs(created_before)); + } + if let Some(modified_after) = search_params.modified_after { + searcher = + searcher.modified_after(SystemTime::UNIX_EPOCH + Duration::from_secs(modified_after)); + } + if let Some(modified_before) = search_params.modified_before { + searcher = + searcher.modified_before(SystemTime::UNIX_EPOCH + Duration::from_secs(modified_before)); + } + + if search_params.hidden { + searcher = searcher.hidden(); + } + if search_params.ignore_case { + searcher = searcher.ignore_case(); + } + searcher.build().collect() +} + +#[tauri::command] +pub async fn file_search(search_params: SearchParams) -> Result, String> { + Ok(search(search_params) + .iter() + .map(|x| PathBuf::from(x)) + .collect()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/fs.rs b/packages/tauri-plugins/jarvis/src/commands/fs.rs new file mode 100644 index 0000000..cbbeaca --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/fs.rs @@ -0,0 +1,37 @@ +use crate::utils; +use std::path::PathBuf; + +#[tauri::command] +pub async fn path_exists(path: String) -> Result { + let exists = PathBuf::from(path.clone()).exists(); + Ok(exists) +} + +#[tauri::command] +pub async fn decompress_tarball( + path: PathBuf, + destination_folder: PathBuf, + overwrite: bool, +) -> Result { + utils::fs::decompress_tarball(&path, &destination_folder, overwrite).map_err(|e| e.to_string()) +} + +/// Compress a directory into a tarball +/// Both `src_dir` and `dest_file` must be be absolute paths +#[tauri::command] +pub async fn compress_tarball( + src_dir: PathBuf, + dest_file: PathBuf, + overwrite: bool, +) -> Result { + utils::fs::compress_tarball(&src_dir, &dest_file, overwrite).map_err(|e| e.to_string()) +} + +#[tauri::command] +pub async fn unzip( + path: PathBuf, + destination_folder: PathBuf, + overwrite: bool, +) -> Result<(), String> { + utils::fs::unzip(&path, &destination_folder, overwrite).map_err(|err| err.to_string()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/mod.rs b/packages/tauri-plugins/jarvis/src/commands/mod.rs new file mode 100644 index 0000000..d6c2d68 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/mod.rs @@ -0,0 +1,14 @@ +pub mod apps; +pub mod clipboard; +pub mod db; +pub mod dev; +pub mod discovery; +pub mod extension; +pub mod file_search; +pub mod fs; +pub mod path; +pub mod server; +// pub mod storage; +pub mod security; +pub mod system; +pub mod utils; diff --git a/packages/tauri-plugins/jarvis/src/commands/path.rs b/packages/tauri-plugins/jarvis/src/commands/path.rs new file mode 100644 index 0000000..5cb00a5 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/path.rs @@ -0,0 +1,18 @@ +use crate::utils::path; +use tauri::Runtime; + +#[tauri::command] +pub async fn get_default_extensions_dir( + app: tauri::AppHandle, +) -> Result<(), String> { + path::get_default_extensions_dir(&app).map_err(|e| e.to_string())?; + Ok(()) +} + +#[tauri::command] +pub async fn get_default_extensions_storage_dir( + app: tauri::AppHandle, +) -> Result<(), String> { + path::get_default_extensions_storage_dir(&app).map_err(|e| e.to_string())?; + Ok(()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/security.rs b/packages/tauri-plugins/jarvis/src/commands/security.rs new file mode 100644 index 0000000..54c9511 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/security.rs @@ -0,0 +1,17 @@ +#![cfg(target_os = "macos")] +#[tauri::command] +pub async fn verify_auth() -> Result { + Ok(mac_security_rs::verify_auth( + mac_security_rs::AuthPolicy::Biometrics, + )) +} + +#[tauri::command] +pub async fn request_screen_capture_access() -> Result { + Ok(mac_security_rs::request_screen_capture_access()) +} + +#[tauri::command] +pub async fn check_screen_capture_access() -> Result { + Ok(mac_security_rs::preflight_screen_capture_access()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/server.rs b/packages/tauri-plugins/jarvis/src/commands/server.rs new file mode 100644 index 0000000..8d09926 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/server.rs @@ -0,0 +1,73 @@ +use crate::{model::app_state, server::http::Server}; +use std::path::PathBuf; + +#[tauri::command] +pub async fn start_server(server: tauri::State<'_, Server>) -> Result<(), String> { + server.start().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn stop_server(server: tauri::State<'_, Server>) -> Result<(), String> { + server.stop().map_err(|err| err.to_string())?; + Ok(()) +} + +#[tauri::command] +pub async fn restart_server(server: tauri::State<'_, Server>) -> Result<(), String> { + server.stop().map_err(|err| err.to_string())?; + server.start().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn server_is_running(server: tauri::State<'_, Server>) -> Result { + Ok(server.is_running()) +} + +#[tauri::command] +pub async fn get_server_port(server: tauri::State<'_, Server>) -> Result { + Ok(server.port) +} + +// #[tauri::command] +// pub async fn set_dev_extension_folder( +// server: tauri::State<'_, Server>, +// app_state: tauri::State<'_, app_state::AppState>, +// dev_ext_folder: Option, +// ) -> Result<(), String> { +// let mut dev_extension_folder = server.dev_extension_folder.lock().unwrap(); +// *dev_extension_folder = dev_ext_folder.clone(); +// let mut app_state_dev_ext_path = app_state.dev_extension_path.lock().unwrap(); +// *app_state_dev_ext_path = dev_ext_folder.clone(); +// Ok(()) +// } + +// #[tauri::command] +// pub async fn set_extension_folder( +// server: tauri::State<'_, Server>, +// app_state: tauri::State<'_, app_state::AppState>, +// ext_folder: PathBuf, +// ) -> Result<(), String> { +// let mut extension_folder = server.extension_folder.lock().unwrap(); +// *extension_folder = ext_folder.clone(); +// let mut extension_path = app_state.extension_path.lock().unwrap(); +// *extension_path = ext_folder; +// Ok(()) +// } + +// #[tauri::command] +// pub async fn get_extension_folder( +// app_state: tauri::State<'_, app_state::AppState>, +// // server: tauri::State<'_, Server>, +// ) -> Result { +// Ok(app_state.extension_path.lock().unwrap().to_owned()) +// // Ok(server.extension_folder.lock().unwrap().to_owned()) +// } + +// #[tauri::command] +// pub async fn get_dev_extension_folder( +// app_state: tauri::State<'_, app_state::AppState>, +// // server: tauri::State<'_, Server>, +// ) -> Result, String> { +// Ok(app_state.dev_extension_path.lock().unwrap().to_owned()) +// // Ok(server.dev_extension_folder.lock().unwrap().to_owned()) +// } diff --git a/packages/tauri-plugins/jarvis/src/commands/storage.rs b/packages/tauri-plugins/jarvis/src/commands/storage.rs new file mode 100644 index 0000000..cd84f97 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/storage.rs @@ -0,0 +1,209 @@ +// /// This file contains a wrapper around the tauri_plugin_store plugin. Instead of using path provided by user, we set a path for each store. +// /// This store is used by extensions, so I don't want them to access store files randomly. +// pub use serde_json::Value as JsonValue; +// use std::path::{Path, PathBuf}; +// use tauri::{AppHandle, Manager, Runtime, State, Window}; +// // use tauri_plugin_store::{Result, Error}; +// type Result = std::result::Result; +// use tauri_plugin_store::{with_store, Store, StoreCollection}; + +// use crate::{utils::path::get_default_extensions_storage_dir, JarvisState}; + +// fn setup_ext_storage_folder( +// handle: &AppHandle, +// identifier: String, +// ) -> anyhow::Result { +// let path = get_default_extensions_storage_dir(handle)?; +// let path = path.join(identifier); +// if !path.exists() { +// std::fs::create_dir_all(&path)?; +// } +// Ok(path) +// } + +// fn preprocess( +// app: &AppHandle, +// window: &Window, +// jarvis_state: &State<'_, JarvisState>, +// path: &PathBuf, +// ) -> Result { +// let window_label = window.label().to_string(); +// let map = jarvis_state.window_label_ext_map.lock().unwrap(); +// // if window_label doesn't start with ext, return error +// if !window_label.starts_with("ext") { +// return Err(format!("Invalid extension window label: {}", window_label)); +// } +// match map.get(window_label.as_str()) { +// Some(ext) => { +// let ext_folder = setup_ext_storage_folder(&app, ext.identifier.clone()) +// .map_err(|e| e.to_string())?; +// let store_path = ext_folder.join(path.file_name().unwrap()); +// Ok(store_path) +// } +// None => { +// return Err(format!("Extension not found for window {}", window_label)); +// } +// } +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_set( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// key: String, +// value: JsonValue, +// ) -> Result<()> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| store.insert(key, value)).map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_get( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// key: String, +// ) -> Result> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| Ok(store.get(key).cloned())) +// .map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_has( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// key: String, +// ) -> Result { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| Ok(store.has(key))).map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_delete( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// key: String, +// ) -> Result { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| store.delete(key)).map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_clear( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// ) -> Result<()> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| store.clear()).map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_reset( +// app: AppHandle, +// window: Window, +// jarvis_state: State<'_, JarvisState>, +// collection: State<'_, StoreCollection>, +// path: PathBuf, +// ) -> Result<()> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, collection, store_path, |store| store.reset()).map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_keys( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// ) -> Result> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| { +// Ok(store.keys().cloned().collect()) +// }) +// .map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_values( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// ) -> Result> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| { +// Ok(store.values().cloned().collect()) +// }) +// .map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_entries( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// ) -> Result> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| { +// Ok(store +// .entries() +// .map(|(k, v)| (k.to_owned(), v.to_owned())) +// .collect()) +// }) +// .map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_length( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// ) -> Result { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| Ok(store.len())).map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_load( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// ) -> Result<()> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| store.load()).map_err(|e| e.to_string()) +// } + +// #[tauri::command] +// pub async fn ext_store_wrapper_save( +// app: AppHandle, +// window: Window, +// stores: State<'_, StoreCollection>, +// jarvis_state: State<'_, JarvisState>, +// path: PathBuf, +// ) -> Result<()> { +// let store_path = preprocess(&app, &window, &jarvis_state, &path).map_err(|e| e.to_string())?; +// with_store(app, stores, store_path, |store| store.save()).map_err(|e| e.to_string()) +// } diff --git a/packages/tauri-plugins/jarvis/src/commands/system.rs b/packages/tauri-plugins/jarvis/src/commands/system.rs new file mode 100644 index 0000000..9c2e386 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/system.rs @@ -0,0 +1,158 @@ +use crate::syscmds::{CommonSystemCmds, SystemCmds}; +use crate::utils::script::run_apple_script; +use applications::AppInfo; + +#[tauri::command] +pub async fn open_trash() -> Result<(), String> { + SystemCmds::open_trash().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn empty_trash() -> Result<(), String> { + SystemCmds::empty_trash().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn shutdown() -> Result<(), String> { + SystemCmds::shutdown().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn reboot() -> Result<(), String> { + SystemCmds::reboot().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn sleep() -> Result<(), String> { + SystemCmds::sleep().map_err(|err| err.to_string()) +} + +/// ```applescript +/// tell application "System Events" to tell appearance preferences +/// set dark_mode to dark mode +/// set dark_mode to not dark_mode +/// set dark mode to dark_mode +/// end tell +/// ``` +#[tauri::command] +pub async fn toggle_system_appearance() -> Result<(), String> { + run_apple_script( + r#" + tell application "System Events" to tell appearance preferences + set dark_mode to dark mode + set dark_mode to not dark_mode + set dark mode to dark_mode + end tell + "#, + ) + .map_err(|err| err.to_string())?; + Ok(()) +} + +#[tauri::command] +pub async fn show_desktop() -> Result<(), String> { + run_apple_script("tell application \"System Events\" to key code 103") + .map_err(|err| err.to_string())?; + Ok(()) +} + +#[tauri::command] +pub async fn quit_all_apps() -> Result<(), String> { + todo!() +} + +#[tauri::command] +pub async fn sleep_displays() -> Result<(), String> { + run_apple_script("do shell script \"pmset displaysleepnow\"").map_err(|err| err.to_string())?; + Ok(()) +} + +/// Set Volume to 0 +#[tauri::command] +pub async fn set_volume(percentage: u8) -> Result<(), String> { + SystemCmds::set_volume(percentage).map_err(|err| err.to_string()) +} +/// Turn Volume Up +#[tauri::command] +pub async fn turn_volume_up() -> Result<(), String> { + SystemCmds::turn_volume_up().map_err(|err| err.to_string()) +} +/// Turn Volume Down +#[tauri::command] +pub async fn turn_volume_down() -> Result<(), String> { + SystemCmds::turn_volume_down().map_err(|err| err.to_string()) +} +/// Toggle Stage Manager +#[tauri::command] +pub async fn toggle_stage_manager() -> Result<(), String> { + todo!() +} + +/// Toggle Bluetooth +#[tauri::command] +pub async fn toggle_bluetooth() -> Result<(), String> { + todo!() +} + +/// Toggle Hidden Files +#[tauri::command] +pub async fn toggle_hidden_files() -> Result<(), String> { + todo!() +} + +/// Eject All Disks +#[tauri::command] +pub async fn eject_all_disks() -> Result<(), String> { + run_apple_script("tell application \"Finder\" to eject (every disk whose ejectable is true)") + .map_err(|err| err.to_string())?; + Ok(()) +} +/// Logout +#[tauri::command] +pub async fn logout_user() -> Result<(), String> { + SystemCmds::logout_user().map_err(|err| err.to_string()) +} +/// Toggle Mute +#[tauri::command] +pub async fn toggle_mute() -> Result<(), String> { + SystemCmds::toggle_mute().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn mute() -> Result<(), String> { + SystemCmds::mute().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn unmute() -> Result<(), String> { + SystemCmds::unmute().map_err(|err| err.to_string()) +} + +#[tauri::command] +pub async fn get_frontmost_app() -> Result { + let ctx = applications::AppInfoContext::new(); + ctx.get_frontmost_application() + .map_err(|err| err.to_string()) +} + +/// Hide All Apps Except Frontmost +#[tauri::command] +pub async fn hide_all_apps_except_frontmost() -> Result<(), String> { + run_apple_script( + r#"tell application "System Events" + set frontApp to name of first application process whose frontmost is true + set visibleApps to every process whose visible is true and name is not frontApp + repeat with theApp in visibleApps + set visible of theApp to false + end repeat + end tell"#, + ) + .map_err(|err| err.to_string())?; + Ok(()) +} + +/// For macOS, return the selected files in the finder +#[tauri::command] +pub async fn get_selected_files_in_file_explorer() -> Result, String> { + SystemCmds::get_selected_files().map_err(|err| err.to_string()) +} diff --git a/packages/tauri-plugins/jarvis/src/commands/utils.rs b/packages/tauri-plugins/jarvis/src/commands/utils.rs new file mode 100644 index 0000000..2025e13 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/commands/utils.rs @@ -0,0 +1,4 @@ +#[tauri::command] +pub async fn plist_to_json(plist_content: String) -> Result { + crate::utils::plist::plist_to_json(plist_content) +} diff --git a/packages/tauri-plugins/jarvis/src/constants.rs b/packages/tauri-plugins/jarvis/src/constants.rs new file mode 100644 index 0000000..ca927c5 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/constants.rs @@ -0,0 +1,13 @@ +/* -------------------------------------------------------------------------- */ +/* Buildin Extension Identifiers */ +/* -------------------------------------------------------------------------- */ +pub const KUNKUN_CLIPBOARD_EXT_IDENTIFIER: &str = "sh.kunkun.ext.clipboard"; +pub const KUNKUN_QUICK_LINKS_EXT_IDENTIFIER: &str = "sh.kunkun.ext.quick-links"; +pub const KUNKUN_REMOTE_EXT_IDENTIFIER: &str = "sh.kunkun.ext.remote"; +pub const KUNKUN_SCRIPT_CMD_EXT_IDENTIFIER: &str = "sh.kunkun.ext.script-cmd"; +pub const KUNKUN_DEV_EXT_IDENTIFIER: &str = "sh.kunkun.ext.dev"; + +/* -------------------------------------------------------------------------- */ +/* Kunkun Builtin Events */ +/* -------------------------------------------------------------------------- */ +pub const KUNKUN_REFRESH_WORKER_EXTENSION: &str = "kunkun://refresh-dev-extension"; diff --git a/packages/tauri-plugins/jarvis/src/desktop.rs b/packages/tauri-plugins/jarvis/src/desktop.rs new file mode 100644 index 0000000..22da24b --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/desktop.rs @@ -0,0 +1,14 @@ +use serde::de::DeserializeOwned; +use tauri::{plugin::PluginApi, AppHandle, Runtime}; + +pub fn init( + app: &AppHandle, + _api: PluginApi, +) -> crate::Result> { + Ok(Jarvis(app.clone())) +} + +/// Access to the jarvis APIs. +pub struct Jarvis(AppHandle); + +impl Jarvis {} diff --git a/packages/tauri-plugins/jarvis/src/error.rs b/packages/tauri-plugins/jarvis/src/error.rs new file mode 100644 index 0000000..177e8c2 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/error.rs @@ -0,0 +1,21 @@ +use serde::{ser::Serializer, Serialize}; + +pub type Result = std::result::Result; + +#[derive(Debug, thiserror::Error)] +pub enum Error { + #[error(transparent)] + Io(#[from] std::io::Error), + #[cfg(mobile)] + #[error(transparent)] + PluginInvoke(#[from] tauri::plugin::mobile::PluginInvokeError), +} + +impl Serialize for Error { + fn serialize(&self, serializer: S) -> std::result::Result + where + S: Serializer, + { + serializer.serialize_str(self.to_string().as_ref()) + } +} diff --git a/packages/tauri-plugins/jarvis/src/lib.rs b/packages/tauri-plugins/jarvis/src/lib.rs new file mode 100644 index 0000000..ca9dbdb --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/lib.rs @@ -0,0 +1,192 @@ +use commands::discovery::Peers; +use db::JarvisDB; +use model::extension::Extension; +use server::Protocol; +use tauri::{ + plugin::{Builder, TauriPlugin}, + Manager, Runtime, +}; +pub mod commands; +pub mod constants; +pub mod model; +pub mod server; +pub mod setup; +pub mod syscmds; +pub mod utils; +pub use db; +use std::{collections::HashMap, path::PathBuf, sync::Mutex}; +use tauri_plugin_store::StoreBuilder; +use utils::{ + path::{get_default_extensions_dir, get_kunkun_db_path}, + settings::AppSettings, +}; + +#[cfg(desktop)] +mod desktop; +#[cfg(mobile)] +mod mobile; + +// mod commands; +mod error; +mod models; + +pub use error::{Error, Result}; + +#[cfg(desktop)] +use desktop::Jarvis; +#[cfg(mobile)] +use mobile::Jarvis; + +#[derive(Default)] +pub struct JarvisState { + pub window_label_ext_map: Mutex>, +} + +/// Extensions to [`tauri::App`], [`tauri::AppHandle`] and [`tauri::Window`] to access the jarvis APIs. +pub trait JarvisExt { + fn jarvis(&self) -> &Jarvis; +} + +impl> crate::JarvisExt for T { + fn jarvis(&self) -> &Jarvis { + self.state::>().inner() + } +} + +/// Initializes the plugin. +pub fn init() -> TauriPlugin { + Builder::new("jarvis") + .invoke_handler(tauri::generate_handler![ + /* ------------------------------ dev commands ------------------------------ */ + commands::dev::open_devtools, + commands::dev::close_devtools, + commands::dev::is_devtools_open, + commands::dev::toggle_devtools, + commands::dev::app_is_dev, + /* ------------------------------ path commands ----------------------------- */ + commands::path::get_default_extensions_dir, + commands::path::get_default_extensions_storage_dir, + /* ----------------------------- system commands ---------------------------- */ + commands::system::open_trash, + commands::system::empty_trash, + commands::system::shutdown, + commands::system::reboot, + commands::system::sleep, + commands::system::toggle_system_appearance, + commands::system::show_desktop, + commands::system::quit_all_apps, + commands::system::sleep_displays, + commands::system::set_volume, + commands::system::turn_volume_up, + commands::system::turn_volume_down, + commands::system::toggle_stage_manager, + commands::system::toggle_bluetooth, + commands::system::toggle_hidden_files, + commands::system::eject_all_disks, + commands::system::logout_user, + commands::system::toggle_mute, + commands::system::mute, + commands::system::unmute, + commands::system::hide_all_apps_except_frontmost, + commands::system::get_frontmost_app, + commands::system::get_selected_files_in_file_explorer, + /* ------------------------------ applications ------------------------------ */ + commands::apps::get_applications, + commands::apps::refresh_applications_list, + commands::apps::refresh_applications_list_in_bg, + /* ------------------------------- extensions ------------------------------- */ + // commands::extension::load_manifest, + // commands::extension::load_all_extensions, + /* ---------------------------------- utils --------------------------------- */ + commands::fs::path_exists, + /* -------------------------------- security -------------------------------- */ + #[cfg(target_os = "macos")] + commands::security::verify_auth, + #[cfg(target_os = "macos")] + commands::security::request_screen_capture_access, + #[cfg(target_os = "macos")] + commands::security::check_screen_capture_access, + /* --------------------------------- server --------------------------------- */ + commands::server::start_server, + commands::server::stop_server, + commands::server::restart_server, + commands::server::server_is_running, + commands::server::get_server_port, + /* ----------------------------------- fs ----------------------------------- */ + commands::fs::decompress_tarball, + commands::fs::compress_tarball, + commands::fs::unzip, + /* ------------------------------- file search ------------------------------ */ + commands::file_search::file_search, + /* ------------------------------- extensions ------------------------------- */ + commands::extension::is_window_label_registered, + commands::extension::register_extension_window, + commands::extension::unregister_extension_window, + commands::extension::register_extension_spawned_process, + commands::extension::unregister_extension_spawned_process, + commands::extension::get_ext_label_map, + /* ---------------------- extension storage API wrapper --------------------- */ + // commands::storage::ext_store_wrapper_set, + // commands::storage::ext_store_wrapper_get, + // commands::storage::ext_store_wrapper_has, + // commands::storage::ext_store_wrapper_delete, + // commands::storage::ext_store_wrapper_clear, + // commands::storage::ext_store_wrapper_reset, + // commands::storage::ext_store_wrapper_keys, + // commands::storage::ext_store_wrapper_values, + // commands::storage::ext_store_wrapper_entries, + // commands::storage::ext_store_wrapper_length, + // commands::storage::ext_store_wrapper_load, + // commands::storage::ext_store_wrapper_save, + /* -------------------------------- database -------------------------------- */ + commands::db::create_extension, + commands::db::get_all_extensions, + commands::db::get_unique_extension_by_identifier, + commands::db::get_unique_extension_by_path, + commands::db::get_all_extensions_by_identifier, + commands::db::delete_extension_by_path, + commands::db::delete_extension_by_ext_id, + commands::db::create_command, + commands::db::get_command_by_id, + commands::db::get_commands_by_ext_id, + commands::db::delete_command_by_id, + commands::db::update_command_by_id, + commands::db::create_extension_data, + commands::db::get_extension_data_by_id, + commands::db::search_extension_data, + commands::db::delete_extension_data_by_id, + commands::db::update_extension_data_by_id, + /* -------------------------------- Clipboard ------------------------------- */ + commands::clipboard::get_history, + commands::clipboard::add_to_history, + /* -------------------------------------------------------------------------- */ + /* Utils */ + /* -------------------------------------------------------------------------- */ + commands::utils::plist_to_json, + /* -------------------------------------------------------------------------- */ + /* MDNS */ + /* -------------------------------------------------------------------------- */ + commands::discovery::get_peers + ]) + .setup(|app, api| { + // #[cfg(mobile)] + // let jarvis = mobile::init(app, api)?; + #[cfg(desktop)] + let jarvis = desktop::init(app, api)?; + app.manage(jarvis); + utils::setup::setup_app_path(app); + utils::setup::setup_extension_storage(app); + + // manage state so it is accessible by the commands + app.manage(JarvisState::default()); + app.manage(commands::apps::ApplicationsState::default()); + let db_path = get_kunkun_db_path(app)?; + let db_key: Option = None; + app.manage(commands::db::DBState::new(db_path.clone(), db_key.clone())?); + setup::db::setup_db(app)?; + println!("Jarvis Plugin Initialized"); + app.manage(Peers::default()); + Ok(()) + }) + .build() +} diff --git a/packages/tauri-plugins/jarvis/src/mobile.rs b/packages/tauri-plugins/jarvis/src/mobile.rs new file mode 100644 index 0000000..39f8f32 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/mobile.rs @@ -0,0 +1,36 @@ +use serde::de::DeserializeOwned; +use tauri::{ + plugin::{PluginApi, PluginHandle}, + AppHandle, Runtime, +}; + +use crate::models::*; + +#[cfg(target_os = "android")] +const PLUGIN_IDENTIFIER: &str = ""; + +#[cfg(target_os = "ios")] +tauri::ios_plugin_binding!(init_plugin_jarvis); + +// initializes the Kotlin or Swift plugin classes +pub fn init( + _app: &AppHandle, + api: PluginApi, +) -> crate::Result> { + #[cfg(target_os = "android")] + let handle = api.register_android_plugin(PLUGIN_IDENTIFIER, "ExamplePlugin")?; + #[cfg(target_os = "ios")] + let handle = api.register_ios_plugin(init_plugin_jarvis)?; + Ok(Jarvis(handle)) +} + +/// Access to the jarvis APIs. +pub struct Jarvis(PluginHandle); + +impl Jarvis { + pub fn ping(&self, payload: PingRequest) -> crate::Result { + self.0 + .run_mobile_plugin("ping", payload) + .map_err(Into::into) + } +} diff --git a/packages/tauri-plugins/jarvis/src/model/app_state.rs b/packages/tauri-plugins/jarvis/src/model/app_state.rs new file mode 100644 index 0000000..7703af6 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/model/app_state.rs @@ -0,0 +1,7 @@ +use std::{path::PathBuf, sync::Mutex}; + +pub struct AppState { + // pub history: Mutex>, + // pub dev_extension_path: Mutex>, + // pub extension_path: Mutex, +} diff --git a/packages/tauri-plugins/jarvis/src/model/clipboard_history.rs b/packages/tauri-plugins/jarvis/src/model/clipboard_history.rs new file mode 100644 index 0000000..05ffff2 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/model/clipboard_history.rs @@ -0,0 +1,91 @@ +use db::JarvisDB; +use serde::{Deserialize, Serialize}; +use std::{str::FromStr, sync::Mutex}; +use strum_macros::{Display, EnumString}; + +#[derive(Debug, Clone, Serialize, Deserialize, Display, EnumString)] +pub enum ClipboardContentType { + Text, + Html, + Rtf, + Image, + // File, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct Record { + pub timestamp: u128, + pub content_type: ClipboardContentType, + pub value: String, + pub text: String, // plain text representation of the value +} + +pub struct ClipboardHistory { + // pub history: Mutex>, + jarvis_db: Mutex, + clipboard_ext_id: i32, +} + +impl ClipboardHistory { + pub fn new(jarvis_db: JarvisDB, clipboard_ext_id: i32) -> Self { + Self { + jarvis_db: Mutex::new(jarvis_db), + clipboard_ext_id, + } + } +} + +impl ClipboardHistory { + pub fn add_record(&self, record: Record) -> anyhow::Result<()> { + let jdb = self.jarvis_db.lock().unwrap(); + jdb.create_extension_data( + self.clipboard_ext_id, + &record.content_type.to_string(), + &record.value, + Some(&record.text), + )?; + Ok(()) + } + + pub fn get_all_records(&self) -> anyhow::Result> { + let jdb = self.jarvis_db.lock().unwrap(); + let db_records = jdb.search_extension_data( + self.clipboard_ext_id, + false, + None, + None, + None, + None, + None, + None, + None, + None, + None, + )?; + let mut records = vec![]; + for r in db_records { + match crate::utils::time::sqlite_timestamp_to_unix_timestamp(&r.created_at) { + Ok(timestamp) => match ClipboardContentType::from_str(&r.data_type) { + Ok(content_type) => { + records.push(Record { + timestamp: timestamp as u128, + content_type, + value: r.data.unwrap().clone(), // safe to unwrap because we are sure it's not None + text: r.search_text.clone().unwrap_or_default(), + }); + } + Err(e) => { + log::error!("Error converting content type: {}", e); + continue; + } + }, + Err(e) => { + log::error!("Error converting timestamp: {}", e); + continue; + } + } + } + Ok(records) + } +} diff --git a/packages/tauri-plugins/jarvis/src/model/extension.rs b/packages/tauri-plugins/jarvis/src/model/extension.rs new file mode 100644 index 0000000..65da38b --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/model/extension.rs @@ -0,0 +1,11 @@ +use super::manifest::Permissions; +use serde::{Deserialize, Serialize}; +use std::path::PathBuf; + +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Extension { + pub path: PathBuf, + pub processes: Vec, + pub dist: Option, + // pub identifier: String, +} diff --git a/packages/tauri-plugins/jarvis/src/model/manifest.rs b/packages/tauri-plugins/jarvis/src/model/manifest.rs new file mode 100644 index 0000000..49017e2 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/model/manifest.rs @@ -0,0 +1,215 @@ +use serde::Deserialize; +use serde::Serialize; +use serde_json::Value; +use std::fmt::Display; +use std::path::PathBuf; + +pub const MANIFEST_FILE_NAME: &str = "package.json"; + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub enum IconType { + #[default] + Iconify, + AssetPath, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub enum TitleBarStyle { + #[default] + Visible, + Transparent, + Overlay, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "lowercase")] +pub enum WindowTheme { + #[default] + Light, + Dark, +} + +#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "lowercase")] +pub enum OSPlatform { + Windows, + MacOS, + Linux, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct Icon { + pub value: String, + #[serde(rename = "type")] + pub type_field: IconType, +} + +#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "kebab-case")] +pub enum Permissions { + ClipboardRead, + ClipboardWrite, + FsHome, + Shell, +} + +impl Display for Permissions { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match self { + Permissions::ClipboardRead => write!(f, "clipboard-read"), + Permissions::ClipboardWrite => write!(f, "clipboard-write"), + Permissions::FsHome => write!(f, "fs-home"), + Permissions::Shell => write!(f, "shell"), + } + } +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct KunkunExtManifest { + pub name: String, + pub short_description: String, + pub long_description: String, + pub identifier: String, + pub icon: Icon, + pub demo_images: Vec, + pub permissions: Option>, + pub ui_cmds: Vec, + #[serde(default = "Vec::new")] + pub inline_cmds: Vec, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct ExtPackageJson { + pub name: String, + pub version: String, + pub kunkun: KunkunExtManifest, + pub files: Vec, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct ExtPackageJsonExtra { + pub name: String, + pub version: String, + pub kunkun: KunkunExtManifest, + pub files: Vec, + // extra fields + pub ext_path: PathBuf, + pub ext_folder_name: String, +} + +impl ExtPackageJsonExtra { + pub fn from(manifest: ExtPackageJson, ext_path: PathBuf) -> Self { + Self { + name: manifest.name, + version: manifest.version, + kunkun: manifest.kunkun, + files: manifest.files, + ext_folder_name: ext_path.file_name().unwrap().to_str().unwrap().to_string(), + ext_path, + } + } +} + +impl ExtPackageJson { + pub fn load(manifest_path: PathBuf) -> anyhow::Result { + let manifest_str = std::fs::read_to_string(manifest_path).unwrap(); + Ok(serde_json::from_str(&manifest_str)?) + } +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct WindowConfig { + pub center: Option, + pub x: Option, + pub y: Option, + pub width: Option, + pub height: Option, + pub min_width: Option, + pub min_height: Option, + pub max_width: Option, + pub max_height: Option, + pub resizable: Option, + pub title: Option, + pub fullscreen: Option, + pub focus: Option, + pub transparent: Option, + pub maximized: Option, + pub visible: Option, + pub decorations: Option, + pub always_on_top: Option, + pub always_on_bottom: Option, + pub content_protected: Option, + pub skip_taskbar: Option, + pub shadow: Option, + pub theme: Option, + pub title_bar_style: Option, + pub hidden_title: Option, + pub tabbing_identifier: Option, + pub maximizable: Option, + pub minimizable: Option, + pub closable: Option, + pub parent: Option, + pub visible_on_all_workspaces: Option, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct UiCmd { + pub main: String, + pub name: String, + pub description: Option, + pub dev_main: String, + pub window: Option, + pub cmds: Vec, + pub platforms: Option>, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct Cmd { + #[serde(rename = "type")] + pub type_field: String, + pub value: String, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct InlineCmd { + pub main: String, + pub name: String, + pub description: Option, + pub cmds: Vec, + pub platforms: Option>, +} + +// // generate test +// #[cfg(test)] +// mod tests { +// use super::*; + +// #[test] +// fn test_load_extension_manifest() { +// // this test relies on submodule +// let manifest_paths = vec![ +// "../../vendors/extensions/extensions/download-twitter-video/package.json", +// "../../vendors/extensions/extensions/jwt/package.json", +// "../../vendors/extensions/extensions/ip-info/package.json", +// "../../vendors/extensions/extensions/qrcode/package.json", +// "../../vendors/extensions/extensions/video-processor/package.json", +// ]; +// for manifest_path in manifest_paths { +// let abs_path = std::fs::canonicalize(manifest_path).unwrap(); +// println!("{:?}", abs_path); +// let manifest_str = std::fs::read_to_string(abs_path).unwrap(); +// let pkg: ExtPackageJson = serde_json::from_str(&manifest_str).unwrap(); +// println!("{:?}", pkg.files); +// } +// } +// } diff --git a/packages/tauri-plugins/jarvis/src/model/mod.rs b/packages/tauri-plugins/jarvis/src/model/mod.rs new file mode 100644 index 0000000..e6d896e --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/model/mod.rs @@ -0,0 +1,4 @@ +pub mod app_state; +pub mod clipboard_history; +pub mod extension; +pub mod manifest; diff --git a/packages/tauri-plugins/jarvis/src/models.rs b/packages/tauri-plugins/jarvis/src/models.rs new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/models.rs @@ -0,0 +1 @@ + diff --git a/packages/tauri-plugins/jarvis/src/server/grpc/greeter.rs b/packages/tauri-plugins/jarvis/src/server/grpc/greeter.rs new file mode 100644 index 0000000..6037a6d --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/server/grpc/greeter.rs @@ -0,0 +1,27 @@ +use hello_world::greeter_server::Greeter; +use hello_world::{HelloReply, HelloRequest}; +use tonic::{Request, Response, Status}; + +pub mod hello_world { + tonic::include_proto!("helloworld"); // The string specified here must match the proto package name + pub const FILE_DESCRIPTOR_SET: &[u8] = + tonic::include_file_descriptor_set!("helloworld_descriptor"); +} + +#[derive(Debug, Default)] +pub struct MyGreeter {} + +#[tonic::async_trait] +impl Greeter for MyGreeter { + async fn say_hello( + &self, + request: Request, // Accept request of type HelloRequest + ) -> Result, Status> { + println!("Got a request: {:?}", request); + let reply = HelloReply { + message: format!("Hello {}!", request.into_inner().name), // We must use .into_inner() as the fields of gRPC requests and responses are private + }; + + Ok(Response::new(reply)) // Send back our formatted greeting + } +} diff --git a/packages/tauri-plugins/jarvis/src/server/grpc/mod.rs b/packages/tauri-plugins/jarvis/src/server/grpc/mod.rs new file mode 100644 index 0000000..44969c6 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/server/grpc/mod.rs @@ -0,0 +1 @@ +pub mod greeter; diff --git a/packages/tauri-plugins/jarvis/src/server/http.rs b/packages/tauri-plugins/jarvis/src/server/http.rs new file mode 100644 index 0000000..5b38c55 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/server/http.rs @@ -0,0 +1,158 @@ +use super::grpc::greeter::hello_world::greeter_server::GreeterServer; +use super::grpc::greeter::MyGreeter; +/// This module is responsible for controlling the main server +use super::model::ServerState; +use super::Protocol; +use crate::server::tls::{CERT_PEM, KEY_PEM}; +use crate::utils::path::get_default_extensions_dir; +use axum::http::{HeaderValue, Method, StatusCode, Uri}; +use axum::routing::{get, get_service, post}; +use axum_server::tls_rustls::RustlsConfig; +use std::sync::Mutex; +use std::{net::SocketAddr, path::PathBuf, sync::Arc}; +use tauri::AppHandle; +use tonic::transport::Server as TonicServer; +use tower_http::{cors::CorsLayer, services::ServeDir}; + +struct ServerOptions {} + +async fn start_server( + protocol: Protocol, + server_addr: SocketAddr, + app_handle: AppHandle, + shtdown_handle: axum_server::Handle, + options: ServerOptions, +) -> Result<(), Box> { + let greeter = MyGreeter::default(); + let server_state = ServerState { + app_handle: app_handle.clone(), + }; + let reflection_service = tonic_reflection::server::Builder::configure() + .register_encoded_file_descriptor_set( + super::grpc::greeter::hello_world::FILE_DESCRIPTOR_SET, + ) + .build() + .unwrap(); + let grpc_router = TonicServer::builder() + .add_service(reflection_service) + .add_service(GreeterServer::new(greeter)) + .into_router(); + let mut rest_router = axum::Router::new() + .route( + "/refresh-worker-extension", + post(super::rest::refresh_worker_extension), + ) + .route("/info", get(super::rest::get_server_info)) + .layer(CorsLayer::permissive()) + .with_state(server_state); + + async fn fallback(uri: Uri) -> (StatusCode, String) { + println!("No route for {uri}"); + (StatusCode::NOT_FOUND, format!("No route for {uri}")) + } + let combined_router = axum::Router::new() + .merge(grpc_router) + .merge(rest_router) + .layer(CorsLayer::permissive()); + let svr = match protocol { + Protocol::Http => { + axum_server::bind(server_addr) + .handle(shtdown_handle) + .serve(combined_router.into_make_service()) + .await + } + Protocol::Https => { + let manifest_dir = PathBuf::from(env!("CARGO_MANIFEST_DIR")); + println!("manifest_dir: {}", manifest_dir.display()); + let tls_config = RustlsConfig::from_pem(CERT_PEM.to_vec(), KEY_PEM.to_vec()).await?; + // let tls_config = RustlsConfig::from_pem_file( + // manifest_dir.join("self_signed_certs").join("server.crt"), + // manifest_dir.join("self_signed_certs").join("server.key"), + // ) + // .await?; + axum_server::bind_rustls(server_addr, tls_config) + .handle(shtdown_handle) + .serve(combined_router.into_make_service()) + .await + } + }; + Ok(svr?) +} + +pub struct Server { + pub app_handle: AppHandle, + pub shtdown_handle: Arc>>, + pub protocol: Mutex, + pub port: u16, + pub server_handle: Arc>>>, +} + +impl Server { + pub fn new(app_handle: AppHandle, port: u16, protocol: Protocol) -> Self { + Self { + app_handle, + protocol: Mutex::new(protocol), + port, + server_handle: Arc::new(std::sync::Mutex::new(None)), + shtdown_handle: Arc::new(Mutex::new(None)), + } + } + + pub async fn set_server_protocol(&self, protocol: Protocol) { + let mut p = self.protocol.lock().unwrap(); + *p = protocol; + } + + pub fn start(&self) -> Result<(), Box> { + let mut server_handle = self.server_handle.lock().unwrap(); + let mut shtdown_handle = self.shtdown_handle.lock().unwrap(); + if server_handle.is_some() { + return Err("Server is already running".into()); + } + let server_addr: SocketAddr = format!("[::]:{}", self.port).parse()?; + let app_handle = self.app_handle.clone(); + let _shutdown_handle = axum_server::Handle::new(); + *shtdown_handle = Some(_shutdown_handle.clone()); + let protocol = self.protocol.lock().unwrap().clone(); + + *server_handle = Some(tauri::async_runtime::spawn(async move { + match start_server( + protocol, + server_addr, + app_handle, + _shutdown_handle, + ServerOptions {}, + ) + .await + { + Ok(_) => {} + Err(e) => { + eprintln!("Server start error: {}", e); + } + } + })); + Ok(()) + } + + pub fn stop(&self) -> Result<(), Box> { + let mut server_handle = self.server_handle.lock().unwrap(); + let mut shtdown_handle = self.shtdown_handle.lock().unwrap(); + match shtdown_handle.as_ref() { + Some(handle) => { + handle.shutdown(); + } + None => { + return Err("Server is not running".into()); + } + } + shtdown_handle.take(); + // self.shutdown_tx.send(())?; + server_handle.take(); + Ok(()) + } + + pub fn is_running(&self) -> bool { + self.server_handle.lock().unwrap().is_some() + && self.shtdown_handle.lock().unwrap().is_some() + } +} diff --git a/packages/tauri-plugins/jarvis/src/server/mod.rs b/packages/tauri-plugins/jarvis/src/server/mod.rs new file mode 100644 index 0000000..5fa4531 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/server/mod.rs @@ -0,0 +1,25 @@ +pub mod grpc; +pub mod http; +pub mod model; +pub mod rest; +pub mod tls; +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Clone, Serialize, Deserialize)] +#[serde(rename_all = "lowercase")] +pub enum Protocol { + Http, + Https, +} + +impl std::fmt::Display for Protocol { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match self { + Protocol::Http => write!(f, "http"), + Protocol::Https => write!(f, "https"), + } + } +} + +pub const CANDIDATE_PORTS: &[u16; 6] = &[1566, 1567, 1568, 9559, 9560, 9561]; +pub const SERVICE_NAME: &str = "jarvis"; diff --git a/packages/tauri-plugins/jarvis/src/server/model.rs b/packages/tauri-plugins/jarvis/src/server/model.rs new file mode 100644 index 0000000..9820355 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/server/model.rs @@ -0,0 +1,14 @@ +use serde::Serialize; +use tauri::{AppHandle, Runtime}; + +#[derive(Serialize)] +pub struct ServerInfo { + pub service_name: String, + pub service_version: String, +} + +#[derive(Clone)] +pub struct ServerState { + // that holds some api specific state + pub app_handle: AppHandle, +} diff --git a/packages/tauri-plugins/jarvis/src/server/rest.rs b/packages/tauri-plugins/jarvis/src/server/rest.rs new file mode 100644 index 0000000..12019d1 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/server/rest.rs @@ -0,0 +1,28 @@ +use super::model::{ServerInfo, ServerState}; +use crate::constants::KUNKUN_REFRESH_WORKER_EXTENSION; +use axum::extract::State; +use tauri::Emitter; + +/// This file contains REST API endpoints + +pub async fn web_root() -> axum::Json { + axum::Json(serde_json::json!({ + "service": "kunkun" + })) +} + +pub async fn get_server_info(State(state): State) -> axum::Json { + let pkg_info = state.app_handle.package_info(); + axum::Json(ServerInfo { + service_name: pkg_info.name.to_string(), + service_version: pkg_info.version.to_string(), + }) +} + +pub async fn refresh_worker_extension(State(state): State) -> &'static str { + state + .app_handle + .emit(KUNKUN_REFRESH_WORKER_EXTENSION, ()) + .unwrap(); + "OK" +} diff --git a/packages/tauri-plugins/jarvis/src/server/tls.rs b/packages/tauri-plugins/jarvis/src/server/tls.rs new file mode 100644 index 0000000..1db0ed9 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/server/tls.rs @@ -0,0 +1,2 @@ +pub const CERT_PEM: &[u8] = include_bytes!("../../self_signed_certs/cert.pem"); +pub const KEY_PEM: &[u8] = include_bytes!("../../self_signed_certs/key.pem"); diff --git a/packages/tauri-plugins/jarvis/src/setup/db.rs b/packages/tauri-plugins/jarvis/src/setup/db.rs new file mode 100644 index 0000000..0f41305 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/setup/db.rs @@ -0,0 +1,29 @@ +use crate::{ + commands, + constants::{ + KUNKUN_CLIPBOARD_EXT_IDENTIFIER, KUNKUN_DEV_EXT_IDENTIFIER, + KUNKUN_QUICK_LINKS_EXT_IDENTIFIER, KUNKUN_REMOTE_EXT_IDENTIFIER, + KUNKUN_SCRIPT_CMD_EXT_IDENTIFIER, + }, +}; +use tauri::{AppHandle, Manager, Runtime}; + +fn create_ext_if_not_exist(identifier: &str, db: &db::JarvisDB) -> anyhow::Result<()> { + let ext = db.get_unique_extension_by_identifier(identifier)?; + if ext.is_none() { + db.create_extension(identifier, "1.0.0", true, None, None)?; + log::info!("Created extension: {}", identifier) + } + Ok(()) +} + +pub fn setup_db(app_handle: &AppHandle) -> anyhow::Result<()> { + let db = app_handle.state::(); + let db = db.db.lock().unwrap(); + create_ext_if_not_exist(KUNKUN_CLIPBOARD_EXT_IDENTIFIER, &db)?; + create_ext_if_not_exist(KUNKUN_QUICK_LINKS_EXT_IDENTIFIER, &db)?; + create_ext_if_not_exist(KUNKUN_REMOTE_EXT_IDENTIFIER, &db)?; + create_ext_if_not_exist(KUNKUN_SCRIPT_CMD_EXT_IDENTIFIER, &db)?; + create_ext_if_not_exist(KUNKUN_DEV_EXT_IDENTIFIER, &db)?; + Ok(()) +} diff --git a/packages/tauri-plugins/jarvis/src/setup/mod.rs b/packages/tauri-plugins/jarvis/src/setup/mod.rs new file mode 100644 index 0000000..78dda75 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/setup/mod.rs @@ -0,0 +1,3 @@ +pub mod db; +pub mod peer_discovery; +pub mod server; diff --git a/packages/tauri-plugins/jarvis/src/setup/peer_discovery.rs b/packages/tauri-plugins/jarvis/src/setup/peer_discovery.rs new file mode 100644 index 0000000..d062560 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/setup/peer_discovery.rs @@ -0,0 +1,38 @@ +use crate::commands::discovery::Peers; +use mdns_sd::ServiceEvent; +use tauri::{AppHandle, Manager, Runtime}; +use tauri_plugin_network::network::mdns::MdnsService; + +pub fn setup_mdns(my_port: u16) -> anyhow::Result { + let mdns = MdnsService::new("tauri")?; + mdns.register( + "tauridesktop", + &MdnsService::get_default_ips_str(), + my_port, + None, + None, + )?; + Ok(mdns) +} + +pub fn handle_mdns_service_evt( + app_handle: &AppHandle, + rx: mdns_sd::Receiver, +) { + let app_handle = app_handle.clone(); + tauri::async_runtime::spawn(async move { + while let Ok(event) = rx.recv() { + match event { + ServiceEvent::ServiceResolved(info) => { + app_handle.state::().add_peer(info.into()); + } + ServiceEvent::ServiceRemoved(service_type, fullname) => { + app_handle + .state::() + .remove_peer(service_type, fullname); + } + _ => {} + } + } + }); +} diff --git a/packages/tauri-plugins/jarvis/src/setup/server.rs b/packages/tauri-plugins/jarvis/src/setup/server.rs new file mode 100644 index 0000000..0ba1978 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/setup/server.rs @@ -0,0 +1,8 @@ +use crate::server::http::Server; +use tauri::{AppHandle, Manager, Runtime}; + +pub fn setup_server(handle: &AppHandle) -> Result<(), Box> { + let server = handle.state::(); + server.start()?; + Ok(()) +} diff --git a/packages/tauri-plugins/jarvis/src/syscmds/linux.rs b/packages/tauri-plugins/jarvis/src/syscmds/linux.rs new file mode 100644 index 0000000..3234111 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/syscmds/linux.rs @@ -0,0 +1,208 @@ +use super::CommonSystemCmds; +use std::process::Command; + +struct Amixer; +impl Amixer { + fn cmd_exists() -> bool { + Command::new("amixer").arg("--version").output().is_ok() + } + fn set_volume(value: String) -> anyhow::Result<()> { + let output = Command::new("amixer") + .arg("set") + .arg("Master") + .arg(value) + .output()?; + + if output.status.success() { + Ok(()) + } else { + Err(anyhow::anyhow!("Failed to set volume")) + } + } + /// amixer -D pulse set Master 1+ mute + /// cmd is mute, unmute or toggle + fn mute_related_cmd(cmd: &str) -> anyhow::Result<()> { + let output = Command::new("amixer") + .arg("-D") + .arg("pulse") + .arg("set") + .arg("Master") + .arg("1+") + .arg(cmd) + .output()?; + + if output.status.success() { + Ok(()) + } else { + Err(anyhow::anyhow!("Failed to mute")) + } + } +} + +struct Pactl; +impl Pactl { + fn cmd_exists() -> bool { + Command::new("pactl").arg("--version").output().is_ok() + } + fn set_volume(value: String) -> anyhow::Result<()> { + let output = Command::new("pactl") + .arg("set-sink-volume") + .arg("@DEFAULT_SINK@") + .arg(value) + .output()?; + + if output.status.success() { + Ok(()) + } else { + Err(anyhow::anyhow!("Failed to set volume")) + } + } + /// pactl set-sink-mute @DEFAULT_SINK@ true + /// cmd is true, false or toggle + fn mute_related_cmd(cmd: &str) -> anyhow::Result<()> { + let output = Command::new("pactl") + .arg("set-sink-mute") + .arg("@DEFAULT_SINK@") + .arg(cmd) + .output()?; + + if output.status.success() { + Ok(()) + } else { + Err(anyhow::anyhow!("Failed to mute")) + } + } +} + +pub struct SystemCmds; +impl CommonSystemCmds for SystemCmds { + /// Run nautilus trash:// + fn open_trash() -> anyhow::Result<()> { + let output = Command::new("nautilus").arg("trash://").output()?; + match output.status.success() { + true => Ok(()), + false => Err(anyhow::anyhow!("Failed to open trash")), + } + } + + fn empty_trash() -> anyhow::Result<()> { + let output = Command::new("rm") + .arg("-rf") + .arg("~/.local/share/Trash/files/*") + .output()?; + match output.status.success() { + true => Ok(()), + false => Err(anyhow::anyhow!("Failed to empty trash")), + } + } + + fn shutdown() -> anyhow::Result<()> { + let output = Command::new("shutdown").arg("-h").arg("now").output()?; + match output.status.success() { + true => Ok(()), + false => Err(anyhow::anyhow!("Failed to shutdown")), + } + } + + fn reboot() -> anyhow::Result<()> { + let output = Command::new("reboot").output()?; + match output.status.success() { + true => Ok(()), + false => Err(anyhow::anyhow!("Failed to reboot")), + } + } + + fn sleep() -> anyhow::Result<()> { + let output = Command::new("systemctl").arg("suspend").output()?; + match output.status.success() { + true => Ok(()), + false => Err(anyhow::anyhow!("Failed to sleep")), + } + } + + fn set_volume(percentage: u8) -> anyhow::Result<()> { + if Pactl::cmd_exists() { + return Pactl::set_volume(format!("{}%", percentage)); + } else if Amixer::cmd_exists() { + return Amixer::set_volume(format!("{}%", percentage)); + } else { + return Err(anyhow::anyhow!( + "No volume control command found (Only Support pactl and amixer)" + )); + } + } + + fn turn_volume_up() -> anyhow::Result<()> { + if Pactl::cmd_exists() { + return Pactl::set_volume("+10%".to_string()); + } else if Amixer::cmd_exists() { + return Amixer::set_volume("10%+".to_string()); + } else { + return Err(anyhow::anyhow!( + "No volume control command found (Only Support pactl and amixer)" + )); + } + } + + fn turn_volume_down() -> anyhow::Result<()> { + if Pactl::cmd_exists() { + return Pactl::set_volume("-10%".to_string()); + } else if Amixer::cmd_exists() { + return Amixer::set_volume("10%-".to_string()); + } else { + return Err(anyhow::anyhow!( + "No volume control command found (Only Support pactl and amixer)" + )); + } + } + + fn logout_user() -> anyhow::Result<()> { + let output = Command::new("pkill").arg("Xorg").output()?; + match output.status.success() { + true => Ok(()), + false => Err(anyhow::anyhow!("Failed to logout")), + } + } + + /// amixer -D pulse set Master 1+ toggle + fn toggle_mute() -> anyhow::Result<()> { + if Pactl::cmd_exists() { + Pactl::mute_related_cmd("toggle") + } else if Amixer::cmd_exists() { + Amixer::mute_related_cmd("toggle") + } else { + Err(anyhow::anyhow!( + "No volume control command found (Only Support pactl and amixer)" + )) + } + } + /// amixer -D pulse set Master 1+ mute + fn mute() -> anyhow::Result<()> { + if Pactl::cmd_exists() { + Pactl::mute_related_cmd("true") + } else if Amixer::cmd_exists() { + Amixer::mute_related_cmd("mute") + } else { + Err(anyhow::anyhow!( + "No volume control command found (Only Support pactl and amixer)" + )) + } + } + + /// amixer -D pulse set Master 1+ unmute + fn unmute() -> anyhow::Result<()> { + if Pactl::cmd_exists() { + Pactl::mute_related_cmd("false") + } else if Amixer::cmd_exists() { + Amixer::mute_related_cmd("unmute") + } else { + Err(anyhow::anyhow!( + "No volume control command found (Only Support pactl and amixer)" + )) + } + } + + fn get_selected_files() -> anyhow::Result> { + Ok(vec![]) + } +} diff --git a/packages/tauri-plugins/jarvis/src/syscmds/mac.rs b/packages/tauri-plugins/jarvis/src/syscmds/mac.rs new file mode 100644 index 0000000..adfd610 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/syscmds/mac.rs @@ -0,0 +1,107 @@ +use super::CommonSystemCmds; +use crate::utils::script::run_apple_script; +use std::path::PathBuf; + +pub struct SystemCmds; + +impl CommonSystemCmds for SystemCmds { + fn open_trash() -> anyhow::Result<()> { + run_apple_script("tell application \"Finder\" to open trash")?; + Ok(()) + } + + fn empty_trash() -> anyhow::Result<()> { + run_apple_script("tell application \"Finder\" to empty the trash")?; + Ok(()) + } + + fn shutdown() -> anyhow::Result<()> { + run_apple_script("tell application \"System Events\" to shut down")?; + Ok(()) + } + + fn reboot() -> anyhow::Result<()> { + run_apple_script("tell application \"System Events\" to restart")?; + Ok(()) + } + + fn sleep() -> anyhow::Result<()> { + run_apple_script("tell application \"System Events\" to sleep")?; + Ok(()) + } + + fn set_volume(percentage: u8) -> anyhow::Result<()> { + run_apple_script(&format!("set volume output volume {}", percentage))?; + Ok(()) + } + + fn turn_volume_up() -> anyhow::Result<()> { + run_apple_script("set volume output volume (output volume of (get volume settings) + 10)")?; + Ok(()) + } + + fn turn_volume_down() -> anyhow::Result<()> { + run_apple_script("set volume output volume (output volume of (get volume settings) - 10)")?; + Ok(()) + } + + fn logout_user() -> anyhow::Result<()> { + run_apple_script("tell application \"System Events\" to log out")?; + Ok(()) + } + + fn toggle_mute() -> anyhow::Result<()> { + run_apple_script( + r#" + -- set volume output muted not (output muted of (get volume settings)) + set curVolume to get volume settings + if output muted of curVolume is false then + set volume with output muted + else + set volume without output muted + end if + "#, + )?; + Ok(()) + } + + fn mute() -> anyhow::Result<()> { + run_apple_script("set volume with output muted")?; + Ok(()) + } + + fn unmute() -> anyhow::Result<()> { + run_apple_script("set volume without output muted")?; + Ok(()) + } + + fn get_selected_files() -> anyhow::Result> { + let stdout = run_apple_script( + r#"use framework "Foundation" + use scripting additions + + tell application "Finder" + set selectedFiles to selection + if (count of selectedFiles) is 0 then + do shell script "echo No files are selected." + else + set filePathList to {} + repeat with i from 1 to count of selectedFiles + set thisFile to item i of selectedFiles + set end of filePathList to (POSIX path of (thisFile as alias)) + end repeat + + -- Convert the list to a JSON string + set jsonArray to current application's NSJSONSerialization's dataWithJSONObject:filePathList options:0 |error|:(missing value) + set jsonString to (current application's NSString's alloc()'s initWithData:jsonArray encoding:(current application's NSUTF8StringEncoding)) + + -- Print the JSON string to stdout + do shell script "echo " & quoted form of (jsonString as text) + end if + end tell + "#, + )?; + let paths: Vec = serde_json::from_str(&stdout)?; + Ok(paths) + } +} diff --git a/packages/tauri-plugins/jarvis/src/syscmds/mod.rs b/packages/tauri-plugins/jarvis/src/syscmds/mod.rs new file mode 100644 index 0000000..f5b15d6 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/syscmds/mod.rs @@ -0,0 +1,30 @@ +#[cfg(target_os = "linux")] +mod linux; +#[cfg(target_os = "linux")] +pub use linux::SystemCmds; +#[cfg(target_os = "macos")] +mod mac; +#[cfg(target_os = "macos")] +pub use mac::SystemCmds; +#[cfg(target_os = "windows")] +mod windows; +#[cfg(target_os = "windows")] +pub use windows::SystemCmds; + +use anyhow::Result; + +pub trait CommonSystemCmds { + fn open_trash() -> Result<()>; + fn empty_trash() -> Result<()>; + fn shutdown() -> Result<()>; + fn reboot() -> Result<()>; + fn sleep() -> Result<()>; + fn set_volume(percentage: u8) -> Result<()>; + fn turn_volume_up() -> Result<()>; + fn turn_volume_down() -> Result<()>; + fn logout_user() -> Result<()>; + fn toggle_mute() -> Result<()>; + fn mute() -> Result<()>; + fn unmute() -> Result<()>; + fn get_selected_files() -> Result>; +} diff --git a/packages/tauri-plugins/jarvis/src/syscmds/windows.rs b/packages/tauri-plugins/jarvis/src/syscmds/windows.rs new file mode 100644 index 0000000..046d2f2 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/syscmds/windows.rs @@ -0,0 +1,130 @@ +use super::CommonSystemCmds; +use crate::utils::script::{run_apple_script, run_powershell}; +pub struct SystemCmds; + +const SET_VOLUME_PS_FUNCTION: &str = r#" + Function Set-Speaker($Volume) { + $wshShell = new-object -com wscript.shell; + 1..50 | % { $wshShell.SendKeys([char]174) }; + if ($Volume -eq 0) { + return + } + 1..$Volume | % { $wshShell.SendKeys([char]175) } + }"#; + +impl CommonSystemCmds for SystemCmds { + fn open_trash() -> anyhow::Result<()> { + run_powershell( + r#" + $shell = New-Object -ComObject Shell.Application + $recycleBin = $shell.Namespace(10) + $recycleBin.Self.InvokeVerb("open") + "#, + )?; + Ok(()) + } + + fn empty_trash() -> anyhow::Result<()> { + run_powershell("Clear-RecycleBin -Force")?; + Ok(()) + } + + fn shutdown() -> anyhow::Result<()> { + run_powershell("Stop-Computer -Force")?; + Ok(()) + } + + fn reboot() -> anyhow::Result<()> { + run_powershell("Restart-Computer -Force")?; + Ok(()) + } + + fn sleep() -> anyhow::Result<()> { + run_powershell("rundll32.exe powrprof.dll,SetSuspendState 0,1,0")?; + Ok(()) + } + + fn set_volume(percentage: u8) -> anyhow::Result<()> { + run_powershell(&format!( + "{}; Set-Speaker {}", + SET_VOLUME_PS_FUNCTION, + percentage / 2 + ))?; + Ok(()) + } + + fn turn_volume_up() -> anyhow::Result<()> { + for _ in 0..5 { + run_powershell( + r#" + $obj = new-object -com wscript.shell + $obj.SendKeys([char]175) + "#, + )?; + } + Ok(()) + } + + fn turn_volume_down() -> anyhow::Result<()> { + for _ in 0..5 { + run_powershell( + r#" + $obj = new-object -com wscript.shell + $obj.SendKeys([char]174) + "#, + )?; + } + Ok(()) + } + + fn logout_user() -> anyhow::Result<()> { + run_powershell("shutdown -l -f")?; + Ok(()) + } + + fn toggle_mute() -> anyhow::Result<()> { + run_powershell( + r#" + $obj = new-object -com wscript.shell + $obj.SendKeys([char]173) + "#, + )?; + Ok(()) + } + + fn mute() -> anyhow::Result<()> { + todo!() + } + + fn unmute() -> anyhow::Result<()> { + todo!() + } + + /// Get the selected files in the Windows File Explorer + fn get_selected_files() -> anyhow::Result> { + let script = r#" + # Create a COM object for the Shell application + $shell = New-Object -ComObject Shell.Application + + # Get all open Windows Explorer windows + $windows = $shell.Windows() + + # Iterate through each window + foreach ($window in $windows) { + # Get the current selection + $selectedItems = $window.Document.SelectedItems() + foreach ($item in $selectedItems) { + # Print the path of each selected file + Write-Output $item.Path + } + } + "#; + let result = run_powershell(script).unwrap(); + let paths: Vec = result + .split('\n') + .map(|path| std::path::PathBuf::from(path.trim())) + .filter(|path| path.exists()) + .collect(); + Ok(paths) + } +} diff --git a/packages/tauri-plugins/jarvis/src/utils/fs.rs b/packages/tauri-plugins/jarvis/src/utils/fs.rs new file mode 100644 index 0000000..1ec7933 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/fs.rs @@ -0,0 +1,119 @@ +use std::fs::File; +use std::path::{Path, PathBuf}; +use zip::ZipArchive; + +/// Decompress a tarball into a destination_folder +/// destination_folder should be a empty folder to avoid being removed +/// The resulting folder is expected to have "package" as its name +/// All .tgz generated with `npm pack` should be decompressed into `package` +pub fn decompress_tarball( + path: &Path, + destination_folder: &Path, + overwrite: bool, +) -> anyhow::Result { + if !path.exists() { + return Err(anyhow::format_err!("Tarball does not exist: {:?}", path)); + } + // if destination exists, remove it + if destination_folder.exists() && overwrite { + std::fs::remove_dir_all(&destination_folder)?; + } + std::fs::create_dir_all(&destination_folder)?; + let tgz = std::fs::File::open(&path)?; + let tar = flate2::read::GzDecoder::new(tgz); + let mut archive = tar::Archive::new(tar); + let dest = destination_folder.join("package"); + + if dest.exists() && !overwrite { + return Err(anyhow::format_err!( + "Destination folder already exists: {:?}", + dest + )); + } + archive.unpack(&destination_folder)?; + if !dest.exists() { + return Err(anyhow::format_err!( + "Failed to unpack tarball to {:?}", + dest + )); + } + Ok(dest) +} + +pub fn compress_tarball( + src_dir: &Path, + dest_file: &Path, + overwrite: bool, +) -> anyhow::Result { + if !src_dir.exists() { + return Err(anyhow::format_err!( + "Source directory does not exist: {:?}", + src_dir + )); + } + if !src_dir.is_dir() { + return Err(anyhow::format_err!( + "Source path is not a directory: {:?}", + src_dir + )); + } + let dest_file = std::fs::canonicalize(dest_file)?; + if dest_file.exists() && !overwrite { + return Err(anyhow::format_err!( + "Destination file already exists: {:?}", + dest_file + )); + } + let tar_gz = std::fs::File::create(&dest_file)?; + let enc = flate2::write::GzEncoder::new(tar_gz, flate2::Compression::default()); + let mut tar = tar::Builder::new(enc); + tar.append_dir_all(src_dir.file_name().unwrap().to_str().unwrap(), &src_dir)?; + Ok(dest_file) +} + +pub fn unzip(path: &Path, destination_folder: &Path, overwrite: bool) -> anyhow::Result<()> { + if destination_folder.exists() && overwrite { + std::fs::remove_dir_all(&destination_folder)?; + } + let file = File::open(path)?; + let mut archive = ZipArchive::new(file)?; + archive.extract(destination_folder)?; + Ok(()) +} + +#[cfg(test)] +mod tests { + // use super::*; + // use std::fs::File; + // use zip::ZipArchive; + // #[test] + // fn test_decompress_tarball() { + // // this test relies on submodule + // decompress_tarball( + // "/Users/hacker/Desktop/huakunshen-jarvis-ext-qrcode-0.0.0.tgz".into(), + // "/Users/hacker/Desktop/randomfolder".into(), + // true, + // ) + // .unwrap(); + // } + + // #[test] + // fn test_unzip() { + // unzip( + // "/Users/hacker/Downloads/bun-darwin-aarch64.zip".into(), + // "/Users/hacker/Downloads/runtime".into(), + // true, + // ) + // .unwrap(); + // } + + // #[test] + // fn test_unzip() -> Result<(), Box> { + // unzip( + // &PathBuf::from("/Users/hacker/Downloads/bun-darwin-aarch64.zip"), + // &PathBuf::from("/Users/hacker/Downloads/runtime"), + // true, + // )?; + // Ok(()) + // } +} diff --git a/packages/tauri-plugins/jarvis/src/utils/icns.rs b/packages/tauri-plugins/jarvis/src/utils/icns.rs new file mode 100644 index 0000000..d33ea81 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/icns.rs @@ -0,0 +1,202 @@ +use applications::utils::image::{self, RustImage, RustImageData}; +use std::{ + ffi::OsStr, + fs::File, + io::{BufReader, Cursor, Write}, + path::{Path, PathBuf}, +}; +#[cfg(target_os = "macos")] +use tauri_icns::{IconFamily, IconType}; +use uuid::Uuid; + +#[cfg(target_os = "macos")] +/// Load Apple icns +pub fn load_icns(icns_path: &PathBuf) -> anyhow::Result { + if icns_path + .extension() + .unwrap_or(OsStr::new("")) + .to_str() + .unwrap() + != "icns" + { + return Err(anyhow::anyhow!("file is not an icns file")); + } + let file = BufReader::new(File::open(icns_path).unwrap()); + let icon_family = IconFamily::read(file).unwrap(); + let mut largest_icon_type = IconType::RGBA32_16x16; + let mut largest_width = 0; + for icon_type in icon_family.available_icons() { + let icon_type_width = icon_type.pixel_width(); + if icon_type_width > largest_width { + largest_width = icon_type_width; + largest_icon_type = icon_type; + if largest_width >= 64 { + // width 64 is large enough + break; + } + } + } + let largest_icon = icon_family.get_icon_with_type(largest_icon_type)?; + let mut buffer: Vec = Vec::new(); + let cursor = Cursor::new(&mut buffer); + largest_icon.write_png(cursor).unwrap(); + let bytes: &[u8] = &buffer; + match RustImageData::from_bytes(bytes) { + Ok(image) => Ok(image), + Err(error) => Err(anyhow::anyhow!(error)), + } +} + +#[cfg(target_os = "linux")] +pub fn load_icon(path: PathBuf) -> tauri::http::Response> { + match path.exists() { + true => { + let bytes = std::fs::read(&path).expect("Error reading file"); + tauri::http::Response::builder().body(bytes).unwrap() + } + false => { + let res = tauri::http::Response::builder() + .status(tauri::http::StatusCode::NOT_FOUND) + .body("file not found".as_bytes().to_vec()) + .unwrap(); + return res; + } + } +} + +#[cfg(target_os = "macos")] +pub fn load_icon(path: PathBuf) -> tauri::http::Response> { + if !path.exists() { + return tauri::http::Response::builder() + .status(tauri::http::StatusCode::NOT_FOUND) + .body("file not found".as_bytes().to_vec()) + .unwrap(); + } + let icns = load_icns(&path); + match icns { + Ok(icns) => { + let png = icns.to_png().unwrap(); + tauri::http::Response::builder() + .body(png.get_bytes().to_vec()) + .unwrap() + } + Err(error) => tauri::http::Response::builder() + .status(tauri::http::StatusCode::INTERNAL_SERVER_ERROR) + .body(error.to_string().as_bytes().to_vec()) + .unwrap(), + } +} + +#[cfg(target_os = "windows")] +pub fn load_icon(path: PathBuf) -> tauri::http::Response> { + // tauri::http::Response::builder().body(vec![]).unwrap() + match path.exists() { + true => { + let ico_loaded = load_ico(&path); + if ico_loaded.is_err() { + let res = tauri::http::Response::builder() + .status(tauri::http::StatusCode::INTERNAL_SERVER_ERROR) + .body("Error loading icon".as_bytes().to_vec()) + .unwrap(); + return res; + } else { + let ico = ico_loaded.unwrap(); + // write ico to random file name.png, read it and return + // Generate a random file name + let id = Uuid::new_v4(); + let file_name = format!("{}.png", id); + // get temp folder + let temp_dir = std::env::temp_dir(); + let file_path = temp_dir.join(file_name); + // Write the ico to the random file name.png + let file = File::create(&file_path).unwrap(); + ico.write_png(file).unwrap(); + // Read the file and return the bytes + let bytes = std::fs::read(&file_path).expect("Error reading file"); + // Delete the file + std::fs::remove_file(&file_path).unwrap(); + tauri::http::Response::builder() + .header("Content-Type", "image/png") + .body(bytes) + .unwrap() + } + } + false => { + let res = tauri::http::Response::builder() + .status(tauri::http::StatusCode::NOT_FOUND) + .body("file not found".as_bytes().to_vec()) + .unwrap(); + return res; + } + } +} + +/// Load .ico image +#[cfg(target_os = "windows")] +pub fn load_ico(path: &Path) -> anyhow::Result { + let file = std::fs::File::open(path)?; + let icon_dir = ico::IconDir::read(file)?; + let image = icon_dir.entries().first(); + if let Some(image) = image { + Ok(image.decode()?) + } else { + Err(anyhow::anyhow!("No image found")) + } +} + +#[cfg(all(test, target_os = "windows"))] +mod tests { + use super::*; + use std::io::Write; + use std::path::PathBuf; + + // #[test] + // fn test_load_icon() { + // let path = PathBuf::from( + // "C:\\Windows\\Installer\\{89C3B1AD-04F9-4A43-940D-51E26BC47942}\\ProductIcon", + // ); + // let icon = load_ico(path).unwrap(); + // icon.write_png(std::fs::File::create("icon.png").unwrap()).unwrap(); + // } +} + +#[cfg(all(test, target_os = "macos"))] +mod tests { + use applications::{utils::mac::MacAppPath, AppTrait}; + + use super::*; + use std::path::PathBuf; + + #[test] + fn test_load_icns() { + let icns_path = + PathBuf::from("/System/Applications/Launchpad.app/Contents/Resources/AppIcon.icns"); + let image = load_icns(&icns_path).unwrap(); + println!("image: {:?}", image.get_size()); + // assert_eq!(image.get_size(), (512, 512)); + // image.save_to_path("icns.png").unwrap(); + } + + #[test] + fn test_app_load_icns() { + let app = MacAppPath::new(PathBuf::from("/Applications/Google Chrome.app")) + .to_app() + .unwrap(); + let icns = app.load_icon().unwrap(); + } + + #[test] + fn debug() { + // let icns_path = + // PathBuf::from("/Applications/Google Chrome.app/Contents/Resources/app.icns"); + // let file = BufReader::new(File::open(icns_path).unwrap()); + // let icon_family = IconFamily::read(file).unwrap(); + // for icon_type in icon_family.available_icons() { + // println!("icon_type: {:?}", icon_type); + // println!("icon_type: {:?}", icon_type.pixel_width()); + // let icon = icon_family.get_icon_with_type(icon_type).unwrap(); + // } + // println!("{:?}", IconType::RGBA32_16x16.fmt()); + // println!("{:?}", IconType::RGB24_16x16.pixel_density()); + } +} diff --git a/packages/tauri-plugins/jarvis/src/utils/manifest.rs b/packages/tauri-plugins/jarvis/src/utils/manifest.rs new file mode 100644 index 0000000..3115322 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/manifest.rs @@ -0,0 +1,58 @@ +use crate::model::manifest::{ExtPackageJson, ExtPackageJsonExtra, MANIFEST_FILE_NAME}; +use anyhow::Result; +use std::path::{Path, PathBuf}; + +pub fn load_jarvis_ext_manifest(manifest_path: PathBuf) -> Result { + // check if it's a folder + + let manifest_path = if manifest_path.is_file() && manifest_path.ends_with(MANIFEST_FILE_NAME) { + manifest_path + } else { + manifest_path.join(MANIFEST_FILE_NAME) + }; + // check if file exists + if !std::path::Path::new(&manifest_path).exists() { + return Err(anyhow::Error::msg(format!( + "{} not found", + manifest_path.to_string_lossy() + ))); + } + ExtPackageJson::load(manifest_path) +} + +pub fn load_all_extensions(extensions_folder: &Path) -> anyhow::Result> { + let mut extensions_with_path: Vec = vec![]; + for entry in std::fs::read_dir(extensions_folder)? { + let entry = entry?; + + if entry.path().join(MANIFEST_FILE_NAME).exists() { + let ext_manifest = load_jarvis_ext_manifest(entry.path()); + if ext_manifest.is_err() { + continue; + } + extensions_with_path.push(ExtPackageJsonExtra::from( + ext_manifest.unwrap(), + entry.path(), + )); + } + } + Ok(extensions_with_path) +} + +// ! Disable test for now as we now load manifest directly with TypeScript +// ! Keeping rust and ts manifest schema in place is troublesome. +// ! In the future, after the schema is stable, and if there is need to load manifest in Rust, we can re-enable this test. +// #[cfg(test)] +// mod test { +// use super::*; + +// #[test] +// fn test_load_jarvis_ext_manifest() { +// let x = PathBuf::from("."); +// let abs_x = x.canonicalize().unwrap(); +// println!("{:?}", abs_x); +// let manifest = +// load_jarvis_ext_manifest(PathBuf::from("../../extensions/hacker-news")).unwrap(); +// assert_eq!(manifest.name, "hacker-news"); +// } +// } diff --git a/packages/tauri-plugins/jarvis/src/utils/mod.rs b/packages/tauri-plugins/jarvis/src/utils/mod.rs new file mode 100644 index 0000000..dc3eccb --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/mod.rs @@ -0,0 +1,9 @@ +pub mod fs; +pub mod icns; +pub mod manifest; +pub mod path; +pub mod plist; +pub mod script; +pub mod settings; +pub mod setup; +pub mod time; diff --git a/packages/tauri-plugins/jarvis/src/utils/path.rs b/packages/tauri-plugins/jarvis/src/utils/path.rs new file mode 100644 index 0000000..e57ad03 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/path.rs @@ -0,0 +1,20 @@ +use std::path::PathBuf; +use tauri::{AppHandle, Manager, Runtime}; + +pub fn get_default_extensions_dir(app: &AppHandle) -> anyhow::Result { + Ok(app.path().app_data_dir()?.join("extensions")) +} + +// pub fn get_default_dev_extensions_dir(app: &AppHandle) -> anyhow::Result { +// Ok(app.path().app_data_dir()?.join("dev-extensions")) +// } + +pub fn get_default_extensions_storage_dir( + app: &AppHandle, +) -> anyhow::Result { + Ok(app.path().app_data_dir()?.join("extensions_storage")) +} + +pub fn get_kunkun_db_path(app: &AppHandle) -> anyhow::Result { + Ok(app.path().app_data_dir()?.join("kk.db")) +} diff --git a/packages/tauri-plugins/jarvis/src/utils/plist.rs b/packages/tauri-plugins/jarvis/src/utils/plist.rs new file mode 100644 index 0000000..0925022 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/plist.rs @@ -0,0 +1,11 @@ +pub fn plist_to_json(plist_content: String) -> Result { + let reader = std::io::Cursor::new(plist_content.as_bytes()); + let parsed_plist_value: plist::Value = + plist::Value::from_reader(reader).expect("Failed to parse plist file"); + // TODO: maybe there is a more elegant way to do this conversion + let json_string = + serde_json::to_string(&parsed_plist_value).expect("Failed to convert plist to json"); + let json_value: serde_json::Value = + serde_json::from_str(&json_string).expect("Failed to convert json to Value"); + Ok(json_value) +} diff --git a/packages/tauri-plugins/jarvis/src/utils/script.rs b/packages/tauri-plugins/jarvis/src/utils/script.rs new file mode 100644 index 0000000..03dfac0 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/script.rs @@ -0,0 +1,31 @@ +use std::process::Command; + +pub fn run_apple_script(script: &str) -> anyhow::Result { + let output = Command::new("osascript") + .arg("-e") + .arg(script) + .output() + .expect("failed to execute applescript"); + match output.status.success() { + true => Ok(String::from_utf8_lossy(&output.stdout).to_string()), + false => Err(anyhow::anyhow!( + "failed to execute applescript: {}", + String::from_utf8_lossy(&output.stderr) + )), + } +} + +pub fn run_powershell(script: &str) -> anyhow::Result { + let output = Command::new("powershell") + .arg("-Command") + .arg(script) + .output() + .expect("failed to execute powershell"); + match output.status.success() { + true => Ok(String::from_utf8_lossy(&output.stdout).to_string()), + false => Err(anyhow::anyhow!( + "failed to execute powershell: {}", + String::from_utf8_lossy(&output.stderr) + )), + } +} diff --git a/packages/tauri-plugins/jarvis/src/utils/settings.rs b/packages/tauri-plugins/jarvis/src/utils/settings.rs new file mode 100644 index 0000000..866730b --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/settings.rs @@ -0,0 +1,57 @@ +use std::path::PathBuf; + +use serde::{Deserialize, Serialize}; +use tauri_plugin_store::Store; + +#[derive(Debug, Clone, Serialize, Deserialize)] +#[serde(rename_all = "lowercase")] +pub enum LightMode { + Auto, + Light, + Dark, +} + +#[derive(Debug, Clone, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct AppSettings { + pub theme: String, + pub radius: f32, + pub light_mode: LightMode, + pub launch_at_login: bool, + pub show_in_tray: bool, + pub dev_extension_path: Option, + pub dev_ext_load_url: bool, + pub hide_on_blur: bool, + pub trigger_hotkey: Option>, +} + +impl Default for AppSettings { + fn default() -> Self { + AppSettings { + theme: "neutral".to_string(), + radius: 0.5, + light_mode: LightMode::Auto, + launch_at_login: true, + show_in_tray: true, + dev_extension_path: None, + dev_ext_load_url: false, + hide_on_blur: true, + trigger_hotkey: None, + } + } +} + +impl AppSettings { + pub fn load_from_store( + store: &Store, + ) -> Result> { + let config = store.get("config"); + match config { + Some(config) => { + let config: AppSettings = serde_json::from_value(config.to_owned())?; + Ok(config) + } + None => Ok(AppSettings::default()), + } + } +} diff --git a/packages/tauri-plugins/jarvis/src/utils/setup.rs b/packages/tauri-plugins/jarvis/src/utils/setup.rs new file mode 100644 index 0000000..a19d51c --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/setup.rs @@ -0,0 +1,47 @@ +// use crate::commands::server::Server; +use tauri::{AppHandle, Manager, Runtime}; + +#[cfg(target_os = "macos")] +use window_vibrancy::{apply_vibrancy, NSVisualEffectMaterial}; + +pub fn setup_app_path(handle: &AppHandle) { + let app_d = handle.path().app_data_dir().unwrap(); + if !app_d.exists() { + std::fs::create_dir_all(&app_d).unwrap(); + } + /* -------------------------------------------------------------------------- */ + /* Create Extensions Folder */ + /* -------------------------------------------------------------------------- */ + let ext_folder = app_d.join("extensions"); + if !ext_folder.exists() { + std::fs::create_dir_all(&ext_folder).unwrap(); + } +} + +// pub fn setup_server(handle: &AppHandle) { +// let server = handle.state::(); +// server.start(handle).expect("Failed to start local server"); +// } + +pub fn setup_mac_transparent_bg(handle: &AppHandle) { + let window = handle.get_webview_window("main").unwrap(); + #[cfg(target_os = "macos")] + apply_vibrancy(&window, NSVisualEffectMaterial::HudWindow, None, None) + .expect("Unsupported platform! 'apply_vibrancy' is only supported on macOS"); +} + +/** + * On MacOS + * app_config: "/Users//Library/Application Support/sh.kunkun" + * app_data: "/Users//Library/Application Support/sh.kunkun" + * app_local_data: "/Users//Library/Application Support/sh.kunkun" + * app_cache: "/Users//Library/Caches/sh.kunkun" + * app_log: "/Users//Library/Logs/sh.kunkun" + */ +pub fn setup_extension_storage(handle: &AppHandle) { + let app_data_dir = handle.path().app_data_dir().unwrap(); + let ext_dir = app_data_dir.join("extensions_support"); + if !ext_dir.exists() { + std::fs::create_dir_all(&ext_dir).unwrap(); + } +} diff --git a/packages/tauri-plugins/jarvis/src/utils/time.rs b/packages/tauri-plugins/jarvis/src/utils/time.rs new file mode 100644 index 0000000..e7d7d28 --- /dev/null +++ b/packages/tauri-plugins/jarvis/src/utils/time.rs @@ -0,0 +1,9 @@ +use chrono::NaiveDateTime; + +pub fn sqlite_timestamp_to_unix_timestamp(timestamp: &str) -> anyhow::Result { + // Parse SQLite timestamp string into a `NaiveDateTime` + let dt = NaiveDateTime::parse_from_str(timestamp, "%Y-%m-%d %H:%M:%S")?; + + // Convert `NaiveDateTime` to Unix timestamp (seconds since epoch) + Ok(dt.and_utc().timestamp()) +} diff --git a/packages/templates/.gitkeep b/packages/templates/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/packages/types/.gitignore b/packages/types/.gitignore new file mode 100644 index 0000000..9b1ee42 --- /dev/null +++ b/packages/types/.gitignore @@ -0,0 +1,175 @@ +# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs + +logs +_.log +npm-debug.log_ +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Caches + +.cache + +# Diagnostic reports (https://nodejs.org/api/report.html) + +report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json + +# Runtime data + +pids +_.pid +_.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover + +lib-cov + +# Coverage directory used by tools like istanbul + +coverage +*.lcov + +# nyc test coverage + +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) + +.grunt + +# Bower dependency directory (https://bower.io/) + +bower_components + +# node-waf configuration + +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) + +build/Release + +# Dependency directories + +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) + +web_modules/ + +# TypeScript cache + +*.tsbuildinfo + +# Optional npm cache directory + +.npm + +# Optional eslint cache + +.eslintcache + +# Optional stylelint cache + +.stylelintcache + +# Microbundle cache + +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history + +.node_repl_history + +# Output of 'npm pack' + +*.tgz + +# Yarn Integrity file + +.yarn-integrity + +# dotenv environment variable files + +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) + +.parcel-cache + +# Next.js build output + +.next +out + +# Nuxt.js build / generate output + +.nuxt +dist + +# Gatsby files + +# Comment in the public line in if your project uses Gatsby and not Next.js + +# https://nextjs.org/blog/next-9-1#public-directory-support + +# public + +# vuepress build output + +.vuepress/dist + +# vuepress v2.x temp and cache directory + +.temp + +# Docusaurus cache and generated files + +.docusaurus + +# Serverless directories + +.serverless/ + +# FuseBox cache + +.fusebox/ + +# DynamoDB Local files + +.dynamodb/ + +# TernJS port file + +.tern-port + +# Stores VSCode versions used for testing VSCode extensions + +.vscode-test + +# yarn v2 + +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# IntelliJ based IDEs +.idea + +# Finder (MacOS) folder config +.DS_Store diff --git a/packages/types/README.md b/packages/types/README.md new file mode 100644 index 0000000..6edaeef --- /dev/null +++ b/packages/types/README.md @@ -0,0 +1,15 @@ +# types + +To install dependencies: + +```bash +bun install +``` + +To run: + +```bash +bun run index.ts +``` + +This project was created using `bun init` in bun v1.1.34. [Bun](https://bun.sh) is a fast all-in-one JavaScript runtime. diff --git a/packages/types/package.json b/packages/types/package.json new file mode 100644 index 0000000..0e35903 --- /dev/null +++ b/packages/types/package.json @@ -0,0 +1,14 @@ +{ + "name": "@kksh/types", + "module": "index.ts", + "type": "module", + "exports": { + ".": "./src/index.ts" + }, + "devDependencies": { + "@types/bun": "latest" + }, + "peerDependencies": { + "typescript": "^5.0.0" + } +} diff --git a/packages/types/src/appConfig.ts b/packages/types/src/appConfig.ts new file mode 100644 index 0000000..31a2dec --- /dev/null +++ b/packages/types/src/appConfig.ts @@ -0,0 +1,28 @@ +import { LightMode } from "@kksh/api/models" +import type { Platform } from "@tauri-apps/plugin-os" +import * as v from "valibot" + +export const PersistedAppConfig = v.object({ + theme: v.object({ + theme: v.string(), + radius: v.number(), + lightMode: LightMode + }), + triggerHotkey: v.nullable(v.array(v.string())), + launchAtLogin: v.boolean(), + showInTray: v.boolean(), + devExtensionPath: v.nullable(v.string()), + hmr: v.boolean(), + hideOnBlur: v.boolean(), + extensionAutoUpgrade: v.boolean(), + joinBetaProgram: v.boolean(), + onBoarded: v.boolean() +}) + +export type PersistedAppConfig = v.InferOutput + +export type AppConfig = PersistedAppConfig & { + isInitialized: boolean + extensionPath?: string + platform: Platform +} diff --git a/packages/types/src/appState.ts b/packages/types/src/appState.ts new file mode 100644 index 0000000..dd9c9f4 --- /dev/null +++ b/packages/types/src/appState.ts @@ -0,0 +1,4 @@ +export interface AppState { + searchTerm: string + highlightedCmd: string +} diff --git a/packages/types/src/index.ts b/packages/types/src/index.ts new file mode 100644 index 0000000..2ed71a9 --- /dev/null +++ b/packages/types/src/index.ts @@ -0,0 +1,2 @@ +export * from "./appConfig" +export * from "./appState" diff --git a/packages/types/tsconfig.json b/packages/types/tsconfig.json new file mode 100644 index 0000000..238655f --- /dev/null +++ b/packages/types/tsconfig.json @@ -0,0 +1,27 @@ +{ + "compilerOptions": { + // Enable latest features + "lib": ["ESNext", "DOM"], + "target": "ESNext", + "module": "ESNext", + "moduleDetection": "force", + "jsx": "react-jsx", + "allowJs": true, + + // Bundler mode + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "noEmit": true, + + // Best practices + "strict": true, + "skipLibCheck": true, + "noFallthroughCasesInSwitch": true, + + // Some stricter flags (disabled by default) + "noUnusedLocals": false, + "noUnusedParameters": false, + "noPropertyAccessFromIndexSignature": false + } +} diff --git a/packages/typescript-config/base.json b/packages/typescript-config/base.json index 30fbd48..4d75084 100644 --- a/packages/typescript-config/base.json +++ b/packages/typescript-config/base.json @@ -7,9 +7,9 @@ "incremental": false, "isolatedModules": true, "lib": ["es2022", "DOM", "DOM.Iterable"], - "module": "NodeNext", + "module": "ESNext", "moduleDetection": "force", - "moduleResolution": "NodeNext", + "moduleResolution": "Bundler", "noUncheckedIndexedAccess": true, "resolveJsonModule": true, "skipLibCheck": true, @@ -17,10 +17,18 @@ "allowImportingTsExtensions": true, "noEmit": true, "target": "ES2022", - "baseUrl": ".", + "baseUrl": "../../", "paths": { - "@kksh/ui/*": ["../../packages/ui/*"], - "@kksh/desktop/*": ["../../packages/desktop/*"] + "@kksh/ui/*": ["./packages/ui/*"], + "@kksh/ui": ["./packages/ui"], + "@kksh/desktop/*": ["./apps/desktop/*"], + "@kksh/desktop": ["./apps/desktop"], + "@kksh/ci/*": ["./packages/ci/*"], + "@kksh/ci": ["./packages/ci"], + "@kksh/supabase/*": ["./packages/supabase/*"], + "@kksh/supabase": ["./packages/supabase"], + "@kksh/schema/*": ["./packages/schema/*"], + "@kksh/schema": ["./packages/schema"] } } } diff --git a/packages/ui/components.json b/packages/ui/components.json index 8c88198..f10b563 100644 --- a/packages/ui/components.json +++ b/packages/ui/components.json @@ -14,4 +14,4 @@ }, "typescript": true, "registry": "https://next.shadcn-svelte.com/registry" -} \ No newline at end of file +} diff --git a/packages/ui/package.json b/packages/ui/package.json index 4690344..4b9e540 100644 --- a/packages/ui/package.json +++ b/packages/ui/package.json @@ -1,31 +1,54 @@ { - "name": "@kksh/ui", - "version": "0.0.1", - "type": "module", - "module": "index.ts", - "main": "index.ts", - "exports": { - ".": { - "types": "./index.ts", - "svelte": "./index.ts" - } - }, - "scripts": { - "lint": "eslint ." - }, - "devDependencies": { - "@kksh/eslint-config": "workspace:*", - "@kksh/typescript-config": "workspace:*", - "bits-ui": "1.0.0-next.36", - "clsx": "^2.1.1", - "lucide-svelte": "^0.454.0", - "mode-watcher": "^0.4.1", - "paneforge": "1.0.0-next.1", - "svelte-radix": "^2.0.1", - "svelte-sonner": "^0.3.28", - "tailwind-merge": "^2.5.4", - "tailwind-variants": "^0.2.1", - "tailwindcss": "^3.4.14", - "tailwindcss-animate": "^1.0.7" - } + "name": "@kksh/ui", + "version": "0.0.1", + "type": "module", + "module": "index.ts", + "main": "index.ts", + "exports": { + ".": { + "types": "./src/index.ts", + "svelte": "./src/index.ts" + }, + "./types": { + "types": "./src/types.ts", + "svelte": "./src/types.ts" + }, + "./custom": { + "types": "./src/components/custom/index.ts", + "svelte": "./src/components/custom/index.ts" + }, + "./utils": { + "types": "./src/utils/index.ts", + "svelte": "./src/utils/index.ts" + }, + "./main": { + "types": "./src/components/main/index.ts", + "svelte": "./src/components/main/index.ts" + }, + "./extension": { + "types": "./src/components/extension/index.ts", + "svelte": "./src/components/extension/index.ts" + } + }, + "scripts": { + "lint": "eslint ." + }, + "devDependencies": { + "@kksh/api": "workspace:*", + "bits-ui": "1.0.0-next.36", + "clsx": "^2.1.1", + "lucide-svelte": "^0.454.0", + "mode-watcher": "^0.4.1", + "paneforge": "1.0.0-next.1", + "shiki": "^1.22.2", + "svelte-radix": "^2.0.1", + "svelte-sonner": "^0.3.28", + "tailwind-merge": "^2.5.4", + "tailwind-variants": "^0.2.1", + "tailwindcss": "^3.4.14", + "tailwindcss-animate": "^1.0.7" + }, + "dependencies": { + "@std/semver": "npm:@jsr/std__semver@^1.0.3" + } } diff --git a/packages/ui/src/components/code/shiki.svelte b/packages/ui/src/components/code/shiki.svelte new file mode 100644 index 0000000..2d663e2 --- /dev/null +++ b/packages/ui/src/components/code/shiki.svelte @@ -0,0 +1,23 @@ + + +
+ {@html html} +
diff --git a/packages/ui/src/components/common/DialogImageCarousel.svelte b/packages/ui/src/components/common/DialogImageCarousel.svelte new file mode 100644 index 0000000..66d9dc0 --- /dev/null +++ b/packages/ui/src/components/common/DialogImageCarousel.svelte @@ -0,0 +1,57 @@ + + + + + + + (api = emblaApi)} class="max-h-[80vh] w-full"> + + {#each imageSrcs as src} + + + + {/each} + + + + + + diff --git a/packages/ui/src/components/common/IconMultiplexer.svelte b/packages/ui/src/components/common/IconMultiplexer.svelte new file mode 100644 index 0000000..7d67db7 --- /dev/null +++ b/packages/ui/src/components/common/IconMultiplexer.svelte @@ -0,0 +1,24 @@ + + +{#if icon.type === IconEnum.RemoteUrl} + +{:else if icon.type === IconEnum.Iconify} + +{:else if icon.type === IconEnum.Base64PNG} + +{:else if icon.type === IconEnum.Text} + +{:else if icon.type === IconEnum.Svg} + {@html icon.value} +{:else} + +{/if} diff --git a/packages/ui/src/components/common/PlatformsIcons.svelte b/packages/ui/src/components/common/PlatformsIcons.svelte new file mode 100644 index 0000000..d3084aa --- /dev/null +++ b/packages/ui/src/components/common/PlatformsIcons.svelte @@ -0,0 +1,27 @@ + + +
+ + + +
diff --git a/packages/ui/src/components/common/index.ts b/packages/ui/src/components/common/index.ts new file mode 100644 index 0000000..9374b63 --- /dev/null +++ b/packages/ui/src/components/common/index.ts @@ -0,0 +1 @@ +export { default as IconMultiplexer } from "./IconMultiplexer.svelte" diff --git a/packages/ui/src/components/custom/draggable-command-group.svelte b/packages/ui/src/components/custom/draggable-command-group.svelte new file mode 100644 index 0000000..67ffa27 --- /dev/null +++ b/packages/ui/src/components/custom/draggable-command-group.svelte @@ -0,0 +1,30 @@ + + + + {#if heading} + + {heading} + + {/if} + + diff --git a/packages/ui/src/components/custom/index.ts b/packages/ui/src/components/custom/index.ts new file mode 100644 index 0000000..8f2d1b3 --- /dev/null +++ b/packages/ui/src/components/custom/index.ts @@ -0,0 +1 @@ +export { default as DraggableCommandGroup } from "./draggable-command-group.svelte" diff --git a/packages/ui/src/components/error/general.svelte b/packages/ui/src/components/error/general.svelte new file mode 100644 index 0000000..5b0d0c4 --- /dev/null +++ b/packages/ui/src/components/error/general.svelte @@ -0,0 +1,34 @@ + + + + + {title} + {message} + + + {@render children?.()} + + + {@render footer?.()} + + diff --git a/packages/ui/src/components/error/index.ts b/packages/ui/src/components/error/index.ts new file mode 100644 index 0000000..b0fcaef --- /dev/null +++ b/packages/ui/src/components/error/index.ts @@ -0,0 +1,2 @@ +export { default as General } from "./general.svelte" +export { default as RawErrorJSONPreset } from "./raw-error-json-preset.svelte" diff --git a/packages/ui/src/components/error/raw-error-json-preset.svelte b/packages/ui/src/components/error/raw-error-json-preset.svelte new file mode 100644 index 0000000..8f13239 --- /dev/null +++ b/packages/ui/src/components/error/raw-error-json-preset.svelte @@ -0,0 +1,55 @@ + + + + + + + + +
+ {#snippet footer()} + {#if footer} + {@render footer()} + {:else} + + {/if} + {/snippet} +
diff --git a/packages/ui/src/components/extension/ExtListItem.svelte b/packages/ui/src/components/extension/ExtListItem.svelte new file mode 100644 index 0000000..3860d65 --- /dev/null +++ b/packages/ui/src/components/extension/ExtListItem.svelte @@ -0,0 +1,76 @@ + + + + + + +
{ext.name}
+ {ext.short_description} +
+
+ + + {#if installedVersion} + {@const upgradable = ext.version + ? greaterThan(parseSemver(ext.version), parseSemver(installedVersion)) + : false} + {#if upgradable} + + {:else} + + {ext.version} + {/if} + {:else} + + {/if} + {humanReadableNumber(ext.downloads)} + +
diff --git a/packages/ui/src/components/extension/PermissionInspector.svelte b/packages/ui/src/components/extension/PermissionInspector.svelte new file mode 100644 index 0000000..44a7557 --- /dev/null +++ b/packages/ui/src/components/extension/PermissionInspector.svelte @@ -0,0 +1,38 @@ + + +
    + {#each manifest?.permissions || [] as perm} +
  • + {typeof perm === "string" ? perm : perm.permission} + + + + + + + + {permissionDescriptions[typeof perm === "string" ? perm : perm.permission]} + + + + + +
  • + {/each} +
diff --git a/packages/ui/src/components/extension/StoreExtDetail.svelte b/packages/ui/src/components/extension/StoreExtDetail.svelte new file mode 100644 index 0000000..b0d9baf --- /dev/null +++ b/packages/ui/src/components/extension/StoreExtDetail.svelte @@ -0,0 +1,195 @@ + + + +{#snippet spinLoader()} + +{/snippet} + +{#snippet upgradeBtn()} + +{/snippet} + +{#snippet uninstallBtn()} + +{/snippet} + +{#snippet installBtn()} + +{/snippet} + +
+ +
+ +
+ + {manifest?.name} + {#if isInstalled} + + {/if} + +
{ext.identifier}
+
Version: {ext.version}
+
+
+ {#if demoImages.length > 0} + + + +
+ {#each demoImages as src, index} + + {/each} +
+
+ {/if} + + +

Security and Privacy

+ + +

Description

+ +
{manifest?.shortDescription}
+
{manifest?.longDescription}
+ +

Commands

+ +
    + {#if manifest} + {#each [...manifest.customUiCmds, ...manifest.templateUiCmds] as cmd} +
  • +
    + {#if manifest} + + {/if} +
    + {cmd.name} +

    {cmd.description}

    +
    + +
    + +
  • + {/each} + {/if} +
+
+ +
+ {#if isInstalled} + {@const isUpgradable = installedExt + ? greaterThan(parseSemver(ext.version), parseSemver(installedExt.version)) + : false} + {#if isUpgradable} +
+ {@render upgradeBtn()} + {@render uninstallBtn()} +
+ {:else}{@render uninstallBtn()}{/if} + {:else} + {@render installBtn()} + {/if} +
diff --git a/packages/ui/src/components/extension/StoreListing.svelte b/packages/ui/src/components/extension/StoreListing.svelte new file mode 100644 index 0000000..05f4b61 --- /dev/null +++ b/packages/ui/src/components/extension/StoreListing.svelte @@ -0,0 +1,59 @@ + + +{#snippet leftSlot()} + +{/snippet} + + + + No results found. + {#each storeExtList as ext} + onExtItemSelected(ext)} + onUpgrade={() => onExtItemUpgrade(ext)} + onInstall={() => onExtItemInstall(ext)} + /> + {/each} + + + diff --git a/packages/ui/src/components/extension/index.ts b/packages/ui/src/components/extension/index.ts new file mode 100644 index 0000000..78a3780 --- /dev/null +++ b/packages/ui/src/components/extension/index.ts @@ -0,0 +1,4 @@ +export { default as ExtListItem } from "./ExtListItem.svelte" +export { default as StoreListing } from "./StoreListing.svelte" +export { default as StoreExtDetail } from "./StoreExtDetail.svelte" +export { default as PermissionInspector } from "./PermissionInspector.svelte" diff --git a/packages/ui/src/components/layouts/center.svelte b/packages/ui/src/components/layouts/center.svelte new file mode 100644 index 0000000..f0d9a00 --- /dev/null +++ b/packages/ui/src/components/layouts/center.svelte @@ -0,0 +1,10 @@ + + +
+ {@render children?.()} +
diff --git a/packages/ui/src/components/layouts/index.ts b/packages/ui/src/components/layouts/index.ts new file mode 100644 index 0000000..a8b4401 --- /dev/null +++ b/packages/ui/src/components/layouts/index.ts @@ -0,0 +1 @@ +export { default as Center } from "./center.svelte" diff --git a/packages/ui/src/components/main/BuiltinCmds.svelte b/packages/ui/src/components/main/BuiltinCmds.svelte new file mode 100644 index 0000000..d4ca49d --- /dev/null +++ b/packages/ui/src/components/main/BuiltinCmds.svelte @@ -0,0 +1,28 @@ + + + + {#each builtinCmds as cmd} + { + cmd.function() + }} + > + + + {cmd.name} + + + {/each} + diff --git a/packages/ui/src/components/main/CustomCommandInput.svelte b/packages/ui/src/components/main/CustomCommandInput.svelte new file mode 100644 index 0000000..bf528b5 --- /dev/null +++ b/packages/ui/src/components/main/CustomCommandInput.svelte @@ -0,0 +1,28 @@ + + +
+ {@render leftSlot?.()} + +
diff --git a/packages/ui/src/components/main/ExtCmdsGroup.svelte b/packages/ui/src/components/main/ExtCmdsGroup.svelte new file mode 100644 index 0000000..59652b9 --- /dev/null +++ b/packages/ui/src/components/main/ExtCmdsGroup.svelte @@ -0,0 +1,60 @@ + + + + +{#snippet cmd(ext: ExtPackageJsonExtra, cmd: CustomUiCmd | TemplateUiCmd)} + { + onExtCmdSelect(ext, cmd, { isDev, hmr }) + }} + > + + + {cmd.name} + + + {#if isDev} + Dev + {/if} + {#if hmr} + HMR + {/if} + + +{/snippet} + +{#snippet ext(ext: ExtPackageJsonExtra)} + {#each ext.kunkun.customUiCmds as _cmd} + {@render cmd(ext, _cmd)} + {/each} + {#each ext.kunkun.templateUiCmds as _cmd} + {@render cmd(ext, _cmd)} + {/each} +{/snippet} + + + {#each extensions as _ext} + {@render ext(_ext)} + {/each} + diff --git a/packages/ui/src/components/main/GlobalCommandPaletteFooter.svelte b/packages/ui/src/components/main/GlobalCommandPaletteFooter.svelte new file mode 100644 index 0000000..6774f30 --- /dev/null +++ b/packages/ui/src/components/main/GlobalCommandPaletteFooter.svelte @@ -0,0 +1,12 @@ + + +
+ + + +
diff --git a/packages/ui/src/components/main/index.ts b/packages/ui/src/components/main/index.ts new file mode 100644 index 0000000..7ff257d --- /dev/null +++ b/packages/ui/src/components/main/index.ts @@ -0,0 +1,5 @@ +export { default as BuiltinCmds } from "./BuiltinCmds.svelte" +export { default as CustomCommandInput } from "./CustomCommandInput.svelte" +export { default as GlobalCommandPaletteFooter } from "./GlobalCommandPaletteFooter.svelte" +export { default as ExtCmdsGroup } from "./ExtCmdsGroup.svelte" +export * from "./types" diff --git a/packages/ui/src/components/main/types.ts b/packages/ui/src/components/main/types.ts new file mode 100644 index 0000000..c8492a1 --- /dev/null +++ b/packages/ui/src/components/main/types.ts @@ -0,0 +1,18 @@ +import type { CustomUiCmd, ExtPackageJsonExtra, TemplateUiCmd } from "@kksh/api/models" + +export type BuiltinCmd = { + name: string + description: string + iconifyIcon: string + function: () => Promise +} + +export type OnExtCmdSelect = ( + ext: ExtPackageJsonExtra, + cmd: CustomUiCmd | TemplateUiCmd, + { isDev, hmr }: { isDev: boolean; hmr: boolean } +) => void + +export type CommandLaunchers = { + onExtCmdSelect: OnExtCmdSelect +} diff --git a/packages/ui/src/index.ts b/packages/ui/src/index.ts new file mode 100644 index 0000000..312b90e --- /dev/null +++ b/packages/ui/src/index.ts @@ -0,0 +1,8 @@ +export { default as Shiki } from "./components/code/shiki.svelte" +export { IconMultiplexer } from "./components/common" +export * as Layouts from "./components/layouts/index" +export * as Error from "./components/error/index" +export * as Common from "./components/common/index" +export * as Custom from "./components/custom" +export * as Main from "./components/main/index" +export * as Extension from "./components/extension/index" diff --git a/packages/ui/src/types.ts b/packages/ui/src/types.ts new file mode 100644 index 0000000..af18f2c --- /dev/null +++ b/packages/ui/src/types.ts @@ -0,0 +1 @@ +export * from "./components/main/types" diff --git a/packages/ui/src/utils/format.ts b/packages/ui/src/utils/format.ts new file mode 100644 index 0000000..f65d5e6 --- /dev/null +++ b/packages/ui/src/utils/format.ts @@ -0,0 +1,19 @@ +/** + * 2500 -> 2.5k + * 2500000 -> 2.5m + * 250,000,000 -> 250m + * 2,500,000,000 -> 2.5b + * @param num + * @returns + */ +export function humanReadableNumber(num: number): string { + if (num < 1000) { + return num.toString() + } else if (num < 1000000) { + return (num / 1000).toFixed(1) + "k" + } else if (num < 1000000000) { + return (num / 1000000).toFixed(1) + "m" + } else { + return (num / 1000000000).toFixed(1) + "b" + } +} diff --git a/packages/ui/src/utils/index.ts b/packages/ui/src/utils/index.ts new file mode 100644 index 0000000..245e021 --- /dev/null +++ b/packages/ui/src/utils/index.ts @@ -0,0 +1,2 @@ +export * from "./tailwind" +export * from "./format" diff --git a/packages/ui/src/utils.ts b/packages/ui/src/utils/tailwind.ts similarity index 100% rename from packages/ui/src/utils.ts rename to packages/ui/src/utils/tailwind.ts diff --git a/packages/ui/tsconfig.json b/packages/ui/tsconfig.json index c678f28..9c56bda 100644 --- a/packages/ui/tsconfig.json +++ b/packages/ui/tsconfig.json @@ -1,7 +1,3 @@ { - "extends": "../typescript-config/base.json", - "compilerOptions": { - "moduleResolution": "Bundler", - "module": "ESNext" - } + "extends": "../typescript-config/base.json" } diff --git a/packages/utils/.gitignore b/packages/utils/.gitignore new file mode 100644 index 0000000..9b1ee42 --- /dev/null +++ b/packages/utils/.gitignore @@ -0,0 +1,175 @@ +# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore + +# Logs + +logs +_.log +npm-debug.log_ +yarn-debug.log* +yarn-error.log* +lerna-debug.log* +.pnpm-debug.log* + +# Caches + +.cache + +# Diagnostic reports (https://nodejs.org/api/report.html) + +report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json + +# Runtime data + +pids +_.pid +_.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover + +lib-cov + +# Coverage directory used by tools like istanbul + +coverage +*.lcov + +# nyc test coverage + +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) + +.grunt + +# Bower dependency directory (https://bower.io/) + +bower_components + +# node-waf configuration + +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) + +build/Release + +# Dependency directories + +node_modules/ +jspm_packages/ + +# Snowpack dependency directory (https://snowpack.dev/) + +web_modules/ + +# TypeScript cache + +*.tsbuildinfo + +# Optional npm cache directory + +.npm + +# Optional eslint cache + +.eslintcache + +# Optional stylelint cache + +.stylelintcache + +# Microbundle cache + +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history + +.node_repl_history + +# Output of 'npm pack' + +*.tgz + +# Yarn Integrity file + +.yarn-integrity + +# dotenv environment variable files + +.env +.env.development.local +.env.test.local +.env.production.local +.env.local + +# parcel-bundler cache (https://parceljs.org/) + +.parcel-cache + +# Next.js build output + +.next +out + +# Nuxt.js build / generate output + +.nuxt +dist + +# Gatsby files + +# Comment in the public line in if your project uses Gatsby and not Next.js + +# https://nextjs.org/blog/next-9-1#public-directory-support + +# public + +# vuepress build output + +.vuepress/dist + +# vuepress v2.x temp and cache directory + +.temp + +# Docusaurus cache and generated files + +.docusaurus + +# Serverless directories + +.serverless/ + +# FuseBox cache + +.fusebox/ + +# DynamoDB Local files + +.dynamodb/ + +# TernJS port file + +.tern-port + +# Stores VSCode versions used for testing VSCode extensions + +.vscode-test + +# yarn v2 + +.yarn/cache +.yarn/unplugged +.yarn/build-state.yml +.yarn/install-state.gz +.pnp.* + +# IntelliJ based IDEs +.idea + +# Finder (MacOS) folder config +.DS_Store diff --git a/packages/utils/README.md b/packages/utils/README.md new file mode 100644 index 0000000..8a1a217 --- /dev/null +++ b/packages/utils/README.md @@ -0,0 +1,15 @@ +# utils + +To install dependencies: + +```bash +bun install +``` + +To run: + +```bash +bun run index.ts +``` + +This project was created using `bun init` in bun v1.1.34. [Bun](https://bun.sh) is a fast all-in-one JavaScript runtime. diff --git a/packages/utils/bun.lockb b/packages/utils/bun.lockb new file mode 100755 index 0000000..79fbdc9 Binary files /dev/null and b/packages/utils/bun.lockb differ diff --git a/packages/utils/package.json b/packages/utils/package.json new file mode 100644 index 0000000..e7f96b6 --- /dev/null +++ b/packages/utils/package.json @@ -0,0 +1,14 @@ +{ + "name": "@kksh/utils", + "module": "index.ts", + "type": "module", + "exports": { + ".": "./src/index.ts" + }, + "devDependencies": { + "@types/bun": "latest" + }, + "peerDependencies": { + "typescript": "^5.0.0" + } +} diff --git a/packages/utils/src/format.ts b/packages/utils/src/format.ts new file mode 100644 index 0000000..e69de29 diff --git a/packages/utils/src/index.ts b/packages/utils/src/index.ts new file mode 100644 index 0000000..4fefa0f --- /dev/null +++ b/packages/utils/src/index.ts @@ -0,0 +1,2 @@ +export * from "./format" +export * from "./time" diff --git a/packages/utils/src/time.ts b/packages/utils/src/time.ts new file mode 100644 index 0000000..f1425bc --- /dev/null +++ b/packages/utils/src/time.ts @@ -0,0 +1,3 @@ +export function sleep(ms: number): Promise { + return new Promise((resolve) => setTimeout(resolve, ms)) +} diff --git a/packages/utils/tsconfig.json b/packages/utils/tsconfig.json new file mode 100644 index 0000000..238655f --- /dev/null +++ b/packages/utils/tsconfig.json @@ -0,0 +1,27 @@ +{ + "compilerOptions": { + // Enable latest features + "lib": ["ESNext", "DOM"], + "target": "ESNext", + "module": "ESNext", + "moduleDetection": "force", + "jsx": "react-jsx", + "allowJs": true, + + // Bundler mode + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "noEmit": true, + + // Best practices + "strict": true, + "skipLibCheck": true, + "noFallthroughCasesInSwitch": true, + + // Some stricter flags (disabled by default) + "noUnusedLocals": false, + "noUnusedParameters": false, + "noPropertyAccessFromIndexSignature": false + } +} diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 3165dc8..4c82edf 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -1,4 +1,4 @@ -lockfileVersion: '6.0' +lockfileVersion: '9.0' settings: autoInstallPeers: true @@ -7,13 +7,89 @@ settings: importers: .: + dependencies: + '@changesets/cli': + specifier: ^2.27.9 + version: 2.27.9 + '@iconify/svelte': + specifier: ^4.0.2 + version: 4.0.2(svelte@5.1.9) + '@supabase/supabase-js': + specifier: ^2.46.1 + version: 2.46.1 + '@tauri-apps/api': + specifier: ^2.0.3 + version: 2.0.3 + '@tauri-apps/cli': + specifier: ^2.0.4 + version: 2.0.4 + '@tauri-apps/plugin-deep-link': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-dialog': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-fs': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-global-shortcut': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-http': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-log': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-notification': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-os': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-process': + specifier: 2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-shell': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-store': + specifier: ^2.1.0 + version: 2.1.0 + '@tauri-apps/plugin-updater': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-upload': + specifier: ^2.0.0 + version: 2.0.0 + supabase: + specifier: ^1.207.9 + version: 1.207.9 + tauri-plugin-network-api: + specifier: workspace:* + version: link:vendors/tauri-plugin-network + tauri-plugin-shellx-api: + specifier: ^2.0.11 + version: 2.0.11 + tauri-plugin-system-info-api: + specifier: workspace:* + version: link:vendors/tauri-plugin-system-info + valibot: + specifier: ^0.40.0 + version: 0.40.0(typescript@5.5.4) + zod: + specifier: ^3.23.8 + version: 3.23.8 devDependencies: '@ianvs/prettier-plugin-sort-imports': specifier: ^4.3.1 - version: 4.3.1(prettier@3.3.3) + version: 4.3.1(@vue/compiler-sfc@3.5.12)(prettier@3.3.3) + '@kksh/api': + specifier: workspace:* + version: link:packages/api '@kksh/svelte5': - specifier: 0.1.2-beta.3 - version: 0.1.2-beta.3(lucide-svelte@0.416.0)(svelte-sonner@0.3.28)(svelte@5.1.9) + specifier: 0.1.2-beta.4 + version: 0.1.2-beta.4(lucide-svelte@0.454.0(svelte@5.1.9))(svelte-sonner@0.3.28(svelte@5.1.9))(svelte@5.1.9) prettier: specifier: ^3.2.5 version: 3.3.3 @@ -22,13 +98,13 @@ importers: version: 3.2.7(prettier@3.3.3)(svelte@5.1.9) prettier-plugin-tailwindcss: specifier: ^0.6.8 - version: 0.6.8(@ianvs/prettier-plugin-sort-imports@4.3.1)(prettier-plugin-svelte@3.2.7)(prettier@3.3.3) + version: 0.6.8(@ianvs/prettier-plugin-sort-imports@4.3.1(@vue/compiler-sfc@3.5.12)(prettier@3.3.3))(prettier-plugin-svelte@3.2.7(prettier@3.3.3)(svelte@5.1.9))(prettier@3.3.3) svelte: specifier: ^5.0.0 version: 5.1.9 svelte-check: specifier: ^4.0.0 - version: 4.0.5(svelte@5.1.9)(typescript@5.5.4) + version: 4.0.5(picomatch@4.0.2)(svelte@5.1.9)(typescript@5.5.4) turbo: specifier: ^2.2.3 version: 2.2.3 @@ -38,25 +114,55 @@ importers: apps/desktop: dependencies: + '@kksh/extension': + specifier: workspace:* + version: link:../../packages/extension + '@kksh/supabase': + specifier: workspace:* + version: link:../../packages/supabase + '@kksh/ui': + specifier: workspace:* + version: link:../../packages/ui + '@kksh/utils': + specifier: workspace:* + version: link:../../packages/utils + '@std/semver': + specifier: npm:@jsr/std__semver@^1.0.3 + version: '@jsr/std__semver@1.0.3' '@tauri-apps/api': specifier: ^2 version: 2.0.3 '@tauri-apps/plugin-shell': specifier: ^2 version: 2.0.1 + bits-ui: + specifier: 1.0.0-next.36 + version: 1.0.0-next.36(svelte@5.1.9) + lucide-svelte: + specifier: ^0.454.0 + version: 0.454.0(svelte@5.1.9) + svelte-radix: + specifier: ^2.0.1 + version: 2.0.1(svelte@5.1.9) + svelte-sonner: + specifier: ^0.3.28 + version: 0.3.28(svelte@5.1.9) + sveltekit-superforms: + specifier: ^2.20.0 + version: 2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3) devDependencies: - '@kksh/ui': + '@kksh/types': specifier: workspace:* - version: link:../../packages/ui + version: link:../../packages/types '@sveltejs/adapter-static': - specifier: ^3.0.5 - version: 3.0.6(@sveltejs/kit@2.7.4) + specifier: ^3.0.6 + version: 3.0.6(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0))) '@sveltejs/kit': - specifier: ^2.7.0 - version: 2.7.4(@sveltejs/vite-plugin-svelte@4.0.0)(svelte@5.1.9)(vite@5.4.10) + specifier: ^2.7.4 + version: 2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) '@sveltejs/vite-plugin-svelte': specifier: ^4.0.0 - version: 4.0.0(svelte@5.1.9)(vite@5.4.10) + version: 4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) '@tailwindcss/aspect-ratio': specifier: ^0.4.2 version: 0.4.2(tailwindcss@3.4.14) @@ -70,7 +176,7 @@ importers: specifier: ^0.5.15 version: 0.5.15(tailwindcss@3.4.14) '@tauri-apps/cli': - specifier: ^2 + specifier: ^2.0.4 version: 2.0.4 autoprefixer: specifier: ^10.4.20 @@ -83,7 +189,7 @@ importers: version: 8.3.1(svelte@5.1.9) formsnap: specifier: ^1.0.1 - version: 1.0.1(svelte@5.1.9)(sveltekit-superforms@2.20.0) + version: 1.0.1(svelte@5.1.9)(sveltekit-superforms@2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)) tailwind-merge: specifier: ^2.5.4 version: 2.5.4 @@ -91,32 +197,269 @@ importers: specifier: ^0.2.1 version: 0.2.1(tailwindcss@3.4.14) tailwindcss: - specifier: ^3.4.9 + specifier: ^3.4.14 version: 3.4.14 tailwindcss-animate: specifier: ^1.0.7 version: 1.0.7(tailwindcss@3.4.14) tslib: - specifier: ^2.8.0 + specifier: ^2.8.1 version: 2.8.1 typescript: - specifier: ^5.5.0 - version: 5.5.4 + specifier: ^5.6.3 + version: 5.6.3 vaul-svelte: specifier: ^0.3.2 version: 0.3.2(svelte@5.1.9) vite: specifier: ^5.4.10 - version: 5.4.10 + version: 5.4.10(@types/node@22.8.7)(terser@5.36.0) + + packages/api: + dependencies: + '@hk/comlink-stdio': + specifier: npm:comlink-stdio@^0.1.7 + version: comlink-stdio@0.1.7(typescript@5.5.4) + '@huakunshen/comlink': + specifier: ^4.4.1 + version: 4.4.1 + '@tauri-apps/api': + specifier: ^2.0.3 + version: 2.0.3 + '@tauri-apps/cli': + specifier: ^2.0.4 + version: 2.0.4 + '@tauri-apps/plugin-deep-link': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-dialog': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-fs': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-global-shortcut': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-http': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-log': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-notification': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-os': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-process': + specifier: 2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-shell': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-store': + specifier: ^2.1.0 + version: 2.1.0 + '@tauri-apps/plugin-updater': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-upload': + specifier: ^2.0.0 + version: 2.0.0 + comlink: + specifier: ^4.4.1 + version: 4.4.1 + lodash: + specifier: ^4.17.21 + version: 4.17.21 + minimatch: + specifier: ^10.0.1 + version: 10.0.1 + semver: + specifier: ^7.6.3 + version: 7.6.3 + svelte-sonner: + specifier: ^0.3.28 + version: 0.3.28(svelte@5.1.9) + tauri-api-adapter: + specifier: 0.3.8 + version: 0.3.8(tslib@2.8.1)(typescript@5.5.4) + tauri-plugin-network-api: + specifier: 2.0.4 + version: 2.0.4(typescript@5.5.4) + tauri-plugin-shellx-api: + specifier: ^2.0.11 + version: 2.0.11 + tauri-plugin-system-info-api: + specifier: 2.0.8 + version: 2.0.8(typescript@5.5.4) + valibot: + specifier: ^0.40.0 + version: 0.40.0(typescript@5.5.4) + devDependencies: + '@types/bun': + specifier: latest + version: 1.1.13 + '@types/lodash': + specifier: ^4.17.13 + version: 4.17.13 + '@types/madge': + specifier: ^5.0.3 + version: 5.0.3 + '@types/node': + specifier: ^22.8.7 + version: 22.8.7 + '@types/semver': + specifier: ^7.5.8 + version: 7.5.8 + fs-extra: + specifier: ^11.2.0 + version: 11.2.0 + madge: + specifier: ^8.0.0 + version: 8.0.0(typescript@5.5.4) + typedoc: + specifier: ^0.26.11 + version: 0.26.11(typescript@5.5.4) + typescript: + specifier: ^5.0.0 + version: 5.5.4 + + packages/api2: + dependencies: + '@hk/comlink-stdio': + specifier: npm:comlink-stdio@^0.1.7 + version: comlink-stdio@0.1.7(typescript@5.6.3) + '@huakunshen/comlink': + specifier: ^4.4.1 + version: 4.4.1 + '@tauri-apps/api': + specifier: ^2.0.3 + version: 2.0.3 + '@tauri-apps/cli': + specifier: ^2.0.4 + version: 2.0.4 + '@tauri-apps/plugin-deep-link': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-dialog': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-fs': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-global-shortcut': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-http': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-log': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-notification': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-os': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-process': + specifier: 2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-shell': + specifier: ^2.0.1 + version: 2.0.1 + '@tauri-apps/plugin-store': + specifier: ^2.1.0 + version: 2.1.0 + '@tauri-apps/plugin-updater': + specifier: ^2.0.0 + version: 2.0.0 + '@tauri-apps/plugin-upload': + specifier: ^2.0.0 + version: 2.0.0 + comlink: + specifier: ^4.4.1 + version: 4.4.1 + lodash: + specifier: ^4.17.21 + version: 4.17.21 + minimatch: + specifier: ^10.0.1 + version: 10.0.1 + semver: + specifier: ^7.6.3 + version: 7.6.3 + svelte-sonner: + specifier: ^0.3.28 + version: 0.3.28(svelte@5.1.9) + tauri-api-adapter: + specifier: 0.3.8 + version: 0.3.8(tslib@2.8.1)(typescript@5.6.3) + tauri-plugin-network-api: + specifier: 2.0.4 + version: 2.0.4(typescript@5.6.3) + tauri-plugin-shellx-api: + specifier: ^2.0.11 + version: 2.0.11 + tauri-plugin-system-info-api: + specifier: 2.0.8 + version: 2.0.8(typescript@5.6.3) + valibot: + specifier: ^0.40.0 + version: 0.40.0(typescript@5.6.3) + devDependencies: + '@types/bun': + specifier: latest + version: 1.1.13 + '@types/lodash': + specifier: ^4.17.13 + version: 4.17.13 + '@types/madge': + specifier: ^5.0.3 + version: 5.0.3 + '@types/node': + specifier: ^22.8.7 + version: 22.8.7 + '@types/semver': + specifier: ^7.5.8 + version: 7.5.8 + fs-extra: + specifier: ^11.2.0 + version: 11.2.0 + madge: + specifier: ^8.0.0 + version: 8.0.0(typescript@5.6.3) + typedoc: + specifier: ^0.26.11 + version: 0.26.11(typescript@5.6.3) + typescript: + specifier: ^5.0.0 + version: 5.6.3 + + packages/ci: + dependencies: + typescript: + specifier: ^5.0.0 + version: 5.5.4 + devDependencies: + '@kksh/typescript-config': + specifier: workspace:* + version: link:../typescript-config + '@types/bun': + specifier: latest + version: 1.1.13 packages/config-eslint: dependencies: '@typescript-eslint/eslint-plugin': specifier: ^7.1.0 - version: 7.18.0(@typescript-eslint/parser@7.18.0)(eslint@8.57.1)(typescript@5.5.4) + version: 7.18.0(@typescript-eslint/parser@7.18.0(eslint@8.57.1)(typescript@5.6.3))(eslint@8.57.1)(typescript@5.6.3) '@typescript-eslint/parser': specifier: ^7.1.0 - version: 7.18.0(eslint@8.57.1)(typescript@5.5.4) + version: 7.18.0(eslint@8.57.1)(typescript@5.6.3) eslint-config-prettier: specifier: ^9.1.0 version: 9.1.0(eslint@8.57.1) @@ -127,16 +470,93 @@ importers: specifier: ^2.35.1 version: 2.46.0(eslint@8.57.1)(svelte@5.1.9) + packages/extension: + dependencies: + '@kksh/api': + specifier: workspace:* + version: link:../api + '@kksh/supabase': + specifier: workspace:* + version: link:../supabase + '@std/semver': + specifier: npm:@jsr/std__semver@^1.0.3 + version: '@jsr/std__semver@1.0.3' + typescript: + specifier: ^5.0.0 + version: 5.5.4 + uuid: + specifier: ^11.0.2 + version: 11.0.2 + devDependencies: + '@types/bun': + specifier: latest + version: 1.1.13 + + packages/schema: + dependencies: + '@aws-sdk/client-s3': + specifier: ^3.583.0 + version: 3.685.0 + '@kksh/api': + specifier: workspace:* + version: link:../api + '@kksh/supabase': + specifier: workspace:* + version: link:../supabase + '@supabase/supabase-js': + specifier: ^2.43.4 + version: 2.46.1 + typescript: + specifier: ^5.0.0 + version: 5.5.4 + valibot: + specifier: ^0.40.0 + version: 0.40.0(typescript@5.5.4) + devDependencies: + '@gcornut/valibot-json-schema': + specifier: ^0.42.0 + version: 0.42.0(esbuild@0.24.0)(typescript@5.5.4) + '@types/bun': + specifier: latest + version: 1.1.13 + supabase: + specifier: '>=1.8.1' + version: 1.207.9 + + packages/supabase: + dependencies: + '@kksh/api': + specifier: workspace:* + version: link:../api + typescript: + specifier: ^5.0.0 + version: 5.5.4 + devDependencies: + '@types/bun': + specifier: latest + version: 1.1.13 + + packages/types: + dependencies: + typescript: + specifier: ^5.0.0 + version: 5.5.4 + devDependencies: + '@types/bun': + specifier: latest + version: 1.1.13 + packages/typescript-config: {} packages/ui: + dependencies: + '@std/semver': + specifier: npm:@jsr/std__semver@^1.0.3 + version: '@jsr/std__semver@1.0.3' devDependencies: - '@kksh/eslint-config': + '@kksh/api': specifier: workspace:* - version: link:../config-eslint - '@kksh/typescript-config': - specifier: workspace:* - version: link:../typescript-config + version: link:../api bits-ui: specifier: 1.0.0-next.36 version: 1.0.0-next.36(svelte@5.1.9) @@ -152,6 +572,9 @@ importers: paneforge: specifier: 1.0.0-next.1 version: 1.0.0-next.1(svelte@5.1.9) + shiki: + specifier: ^1.22.2 + version: 1.22.2 svelte-radix: specifier: ^2.0.1 version: 2.0.1(svelte@5.1.9) @@ -171,517 +594,719 @@ importers: specifier: ^1.0.7 version: 1.0.7(tailwindcss@3.4.14) + packages/utils: + dependencies: + typescript: + specifier: ^5.0.0 + version: 5.5.4 + devDependencies: + '@types/bun': + specifier: latest + version: 1.1.13 + + vendors/tauri-plugin-network: + dependencies: + '@tauri-apps/api': + specifier: ^2.0.1 + version: 2.0.3 + valibot: + specifier: ^0.40.0 + version: 0.40.0(typescript@5.5.4) + devDependencies: + '@rollup/plugin-typescript': + specifier: ^11.1.6 + version: 11.1.6(rollup@4.24.3)(tslib@2.8.1)(typescript@5.5.4) + rollup: + specifier: ^4.24.0 + version: 4.24.3 + tslib: + specifier: ^2.6.2 + version: 2.8.1 + typedoc: + specifier: ^0.26.6 + version: 0.26.11(typescript@5.5.4) + typescript: + specifier: ^5.3.3 + version: 5.5.4 + + vendors/tauri-plugin-system-info: + dependencies: + '@tauri-apps/api': + specifier: ^2.0.1 + version: 2.0.3 + valibot: + specifier: ^0.40.0 + version: 0.40.0(typescript@5.5.4) + devDependencies: + '@rollup/plugin-typescript': + specifier: ^11.1.6 + version: 11.1.6(rollup@4.24.3)(tslib@2.8.1)(typescript@5.5.4) + rollup: + specifier: ^4.24.0 + version: 4.24.3 + tslib: + specifier: ^2.6.2 + version: 2.8.1 + typedoc: + specifier: ^0.26.6 + version: 0.26.11(typescript@5.5.4) + typescript: + specifier: ^5.3.3 + version: 5.5.4 + packages: - /@alloc/quick-lru@5.2.0: + '@alloc/quick-lru@5.2.0': resolution: {integrity: sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==} engines: {node: '>=10'} - dev: true - /@ampproject/remapping@2.3.0: + '@ampproject/remapping@2.3.0': resolution: {integrity: sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==} engines: {node: '>=6.0.0'} - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - /@ark/schema@0.10.0: + '@ark/schema@0.10.0': resolution: {integrity: sha512-zpfXwWLOzj9aUK+dXQ6aleJAOgle4/WrHDop5CMX2M88dFQ85NdH8O0v0pvMAQnfFcaQAZ/nVDYLlBJsFc09XA==} - requiresBuild: true - dependencies: - '@ark/util': 0.10.0 - dev: true - optional: true - /@ark/util@0.10.0: + '@ark/util@0.10.0': resolution: {integrity: sha512-uK+9VU5doGMYOoOZVE+XaSs1vYACoaEJdrDkuBx26S4X7y3ChyKsPnIg/9pIw2vUySph1GkAXbvBnfVE2GmXgQ==} - requiresBuild: true - dev: true - optional: true - /@babel/code-frame@7.26.2: + '@aws-crypto/crc32@5.2.0': + resolution: {integrity: sha512-nLbCWqQNgUiwwtFsen1AdzAtvuLRsQS8rYgMuxCrdKf9kOssamGLuPwyTY9wyYblNr9+1XM8v6zoDTPPSIeANg==} + engines: {node: '>=16.0.0'} + + '@aws-crypto/crc32c@5.2.0': + resolution: {integrity: sha512-+iWb8qaHLYKrNvGRbiYRHSdKRWhto5XlZUEBwDjYNf+ly5SVYG6zEoYIdxvf5R3zyeP16w4PLBn3rH1xc74Rag==} + + '@aws-crypto/sha1-browser@5.2.0': + resolution: {integrity: sha512-OH6lveCFfcDjX4dbAvCFSYUjJZjDr/3XJ3xHtjn3Oj5b9RjojQo8npoLeA/bNwkOkrSQ0wgrHzXk4tDRxGKJeg==} + + '@aws-crypto/sha256-browser@5.2.0': + resolution: {integrity: sha512-AXfN/lGotSQwu6HNcEsIASo7kWXZ5HYWvfOmSNKDsEqC4OashTp8alTmaz+F7TC2L083SFv5RdB+qU3Vs1kZqw==} + + '@aws-crypto/sha256-js@5.2.0': + resolution: {integrity: sha512-FFQQyu7edu4ufvIZ+OadFpHHOt+eSTBaYaki44c+akjg7qZg9oOQeLlk77F6tSYqjDAFClrHJk9tMf0HdVyOvA==} + engines: {node: '>=16.0.0'} + + '@aws-crypto/supports-web-crypto@5.2.0': + resolution: {integrity: sha512-iAvUotm021kM33eCdNfwIN//F77/IADDSs58i+MDaOqFrVjZo9bAal0NK7HurRuWLLpF1iLX7gbWrjHjeo+YFg==} + + '@aws-crypto/util@5.2.0': + resolution: {integrity: sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ==} + + '@aws-sdk/client-s3@3.685.0': + resolution: {integrity: sha512-ClvMeQHbLhWkpxnVymo4qWS5/yZcPXjorDbSday3joCWYWCSHTO409nWd+jx6eA4MKT/EY/uJ6ZBJRFfByKLuA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/client-sso-oidc@3.682.0': + resolution: {integrity: sha512-ZPZ7Y/r/w3nx/xpPzGSqSQsB090Xk5aZZOH+WBhTDn/pBEuim09BYXCLzvvxb7R7NnuoQdrTJiwimdJAhHl7ZQ==} + engines: {node: '>=16.0.0'} + peerDependencies: + '@aws-sdk/client-sts': ^3.682.0 + + '@aws-sdk/client-sso@3.682.0': + resolution: {integrity: sha512-PYH9RFUMYLFl66HSBq4tIx6fHViMLkhJHTYJoJONpBs+Td+NwVJ895AdLtDsBIhMS0YseCbPpuyjUCJgsUrwUw==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/client-sts@3.682.0': + resolution: {integrity: sha512-xKuo4HksZ+F8m9DOfx/ZuWNhaPuqZFPwwy0xqcBT6sWH7OAuBjv/fnpOTzyQhpVTWddlf+ECtMAMrxjxuOExGQ==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/core@3.679.0': + resolution: {integrity: sha512-CS6PWGX8l4v/xyvX8RtXnBisdCa5+URzKd0L6GvHChype9qKUVxO/Gg6N/y43Hvg7MNWJt9FBPNWIxUB+byJwg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/credential-provider-env@3.679.0': + resolution: {integrity: sha512-EdlTYbzMm3G7VUNAMxr9S1nC1qUNqhKlAxFU8E7cKsAe8Bp29CD5HAs3POc56AVo9GC4yRIS+/mtlZSmrckzUA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/credential-provider-http@3.679.0': + resolution: {integrity: sha512-ZoKLubW5DqqV1/2a3TSn+9sSKg0T8SsYMt1JeirnuLJF0mCoYFUaWMyvxxKuxPoqvUsaycxKru4GkpJ10ltNBw==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/credential-provider-ini@3.682.0': + resolution: {integrity: sha512-6eqWeHdK6EegAxqDdiCi215nT3QZPwukgWAYuVxNfJ/5m0/P7fAzF+D5kKVgByUvGJEbq/FEL8Fw7OBe64AA+g==} + engines: {node: '>=16.0.0'} + peerDependencies: + '@aws-sdk/client-sts': ^3.682.0 + + '@aws-sdk/credential-provider-node@3.682.0': + resolution: {integrity: sha512-HSmDqZcBVZrTctHCT9m++vdlDfJ1ARI218qmZa+TZzzOFNpKWy6QyHMEra45GB9GnkkMmV6unoDSPMuN0AqcMg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/credential-provider-process@3.679.0': + resolution: {integrity: sha512-u/p4TV8kQ0zJWDdZD4+vdQFTMhkDEJFws040Gm113VHa/Xo1SYOjbpvqeuFoz6VmM0bLvoOWjxB9MxnSQbwKpQ==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/credential-provider-sso@3.682.0': + resolution: {integrity: sha512-h7IH1VsWgV6YAJSWWV6y8uaRjGqLY3iBpGZlXuTH/c236NMLaNv+WqCBLeBxkFGUb2WeQ+FUPEJDCD69rgLIkg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/credential-provider-web-identity@3.679.0': + resolution: {integrity: sha512-a74tLccVznXCaBefWPSysUcLXYJiSkeUmQGtalNgJ1vGkE36W5l/8czFiiowdWdKWz7+x6xf0w+Kjkjlj42Ung==} + engines: {node: '>=16.0.0'} + peerDependencies: + '@aws-sdk/client-sts': ^3.679.0 + + '@aws-sdk/middleware-bucket-endpoint@3.679.0': + resolution: {integrity: sha512-5EpiPhhGgnF+uJR4DzWUk6Lx3pOn9oM6JGXxeHsiynfoBfq7vHMleq+uABHHSQS+y7XzbyZ7x8tXNQlliMwOsg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-expect-continue@3.679.0': + resolution: {integrity: sha512-nYsh9PdWrF4EahTRdXHGlNud82RPc508CNGdh1lAGfPU3tNveGfMBX3PcGBtPOse3p9ebNKRWVmUc9eXSjGvHA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-flexible-checksums@3.682.0': + resolution: {integrity: sha512-5u1STth6iZUtAvPDO0NJVYKUX2EYKU7v84MYYaZ3O27HphRjFqDos0keL2KTnHn/KmMD68rM3yiUareWR8hnAQ==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-host-header@3.679.0': + resolution: {integrity: sha512-y176HuQ8JRY3hGX8rQzHDSbCl9P5Ny9l16z4xmaiLo+Qfte7ee4Yr3yaAKd7GFoJ3/Mhud2XZ37fR015MfYl2w==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-location-constraint@3.679.0': + resolution: {integrity: sha512-SA1C1D3XgoKTGxyNsOqd016ONpk46xJLWDgJUd00Zb21Ox5wYCoY6aDRKiaMRW+1VfCJdezs1Do3XLyIU9KxyA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-logger@3.679.0': + resolution: {integrity: sha512-0vet8InEj7nvIvGKk+ch7bEF5SyZ7Us9U7YTEgXPrBNStKeRUsgwRm0ijPWWd0a3oz2okaEwXsFl7G/vI0XiEA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-recursion-detection@3.679.0': + resolution: {integrity: sha512-sQoAZFsQiW/LL3DfKMYwBoGjYDEnMbA9WslWN8xneCmBAwKo6IcSksvYs23PP8XMIoBGe2I2J9BSr654XWygTQ==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-sdk-s3@3.685.0': + resolution: {integrity: sha512-C4w92b3A99NbghrA2Ssw6y1RbDF3I3Bgzi2Izh0pXgyIoDiX0xs9bUs/FGYLK4uepYr78DAZY8DwEpzjWIXkSA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-ssec@3.679.0': + resolution: {integrity: sha512-4GNUxXbs1M71uFHRiCAZtN0/g23ogI9YjMe5isAuYMHXwDB3MhqF7usKf954mBP6tplvN44vYlbJ84faaLrTtg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/middleware-user-agent@3.682.0': + resolution: {integrity: sha512-7TyvYR9HdGH1/Nq0eeApUTM4izB6rExiw87khVYuJwZHr6FmvIL1FsOVFro/4WlXa0lg4LiYOm/8H8dHv+fXTg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/region-config-resolver@3.679.0': + resolution: {integrity: sha512-Ybx54P8Tg6KKq5ck7uwdjiKif7n/8g1x+V0V9uTjBjRWqaIgiqzXwKWoPj6NCNkE7tJNtqI4JrNxp/3S3HvmRw==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/signature-v4-multi-region@3.685.0': + resolution: {integrity: sha512-IHLwuAZGqfUWVrNqw0ugnBa7iL8uBP4x6A7bfBDXRXWCWjUCed/1/D//0lKDHwpFkV74fGW6KoBacnWSUlXmwA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/token-providers@3.679.0': + resolution: {integrity: sha512-1/+Zso/x2jqgutKixYFQEGli0FELTgah6bm7aB+m2FAWH4Hz7+iMUsazg6nSWm714sG9G3h5u42Dmpvi9X6/hA==} + engines: {node: '>=16.0.0'} + peerDependencies: + '@aws-sdk/client-sso-oidc': ^3.679.0 + + '@aws-sdk/types@3.679.0': + resolution: {integrity: sha512-NwVq8YvInxQdJ47+zz4fH3BRRLC6lL+WLkvr242PVBbUOLRyK/lkwHlfiKUoeVIMyK5NF+up6TRg71t/8Bny6Q==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/util-arn-parser@3.679.0': + resolution: {integrity: sha512-CwzEbU8R8rq9bqUFryO50RFBlkfufV9UfMArHPWlo+lmsC+NlSluHQALoj6Jkq3zf5ppn1CN0c1DDLrEqdQUXg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/util-endpoints@3.679.0': + resolution: {integrity: sha512-YL6s4Y/1zC45OvddvgE139fjeWSKKPgLlnfrvhVL7alNyY9n7beR4uhoDpNrt5mI6sn9qiBF17790o+xLAXjjg==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/util-locate-window@3.679.0': + resolution: {integrity: sha512-zKTd48/ZWrCplkXpYDABI74rQlbR0DNHs8nH95htfSLj9/mWRSwaGptoxwcihaq/77vi/fl2X3y0a1Bo8bt7RA==} + engines: {node: '>=16.0.0'} + + '@aws-sdk/util-user-agent-browser@3.679.0': + resolution: {integrity: sha512-CusSm2bTBG1kFypcsqU8COhnYc6zltobsqs3nRrvYqYaOqtMnuE46K4XTWpnzKgwDejgZGOE+WYyprtAxrPvmQ==} + + '@aws-sdk/util-user-agent-node@3.682.0': + resolution: {integrity: sha512-so5s+j0gPoTS0HM4HPL+G0ajk0T6cQAg8JXzRgvyiQAxqie+zGCZAV3VuVeMNWMVbzsgZl0pYZaatPFTLG/AxA==} + engines: {node: '>=16.0.0'} + peerDependencies: + aws-crt: '>=1.0.0' + peerDependenciesMeta: + aws-crt: + optional: true + + '@aws-sdk/xml-builder@3.679.0': + resolution: {integrity: sha512-nPmhVZb39ty5bcQ7mAwtjezBcsBqTYZ9A2D9v/lE92KCLdu5RhSkPH7O71ZqbZx1mUSg9fAOxHPiG79U5VlpLQ==} + engines: {node: '>=16.0.0'} + + '@babel/code-frame@7.26.2': resolution: {integrity: sha512-RJlIHRueQgwWitWgF8OdFYGZX328Ax5BCemNGlqHfplnRT9ESi8JkFlvaVYbS+UubVY6dpv87Fs2u5M29iNFVQ==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/helper-validator-identifier': 7.25.9 - js-tokens: 4.0.0 - picocolors: 1.1.0 - dev: true - /@babel/compat-data@7.26.2: + '@babel/compat-data@7.26.2': resolution: {integrity: sha512-Z0WgzSEa+aUcdiJuCIqgujCshpMWgUpgOxXotrYPSA53hA3qopNaqcJpyr0hVb1FeWdnqFA35/fUtXgBK8srQg==} engines: {node: '>=6.9.0'} - dev: true - /@babel/core@7.26.0: + '@babel/core@7.26.0': resolution: {integrity: sha512-i1SLeK+DzNnQ3LL/CswPCa/E5u4lh1k6IAEphON8F+cXt0t9euTshDru0q7/IqMa1PMPz5RnHuHscF8/ZJsStg==} engines: {node: '>=6.9.0'} - dependencies: - '@ampproject/remapping': 2.3.0 - '@babel/code-frame': 7.26.2 - '@babel/generator': 7.26.2 - '@babel/helper-compilation-targets': 7.25.9 - '@babel/helper-module-transforms': 7.26.0(@babel/core@7.26.0) - '@babel/helpers': 7.26.0 - '@babel/parser': 7.26.2 - '@babel/template': 7.25.9 - '@babel/traverse': 7.25.9 - '@babel/types': 7.26.0 - convert-source-map: 2.0.0 - debug: 4.3.7 - gensync: 1.0.0-beta.2 - json5: 2.2.3 - semver: 6.3.1 - transitivePeerDependencies: - - supports-color - dev: true - /@babel/generator@7.26.2: + '@babel/generator@7.26.2': resolution: {integrity: sha512-zevQbhbau95nkoxSq3f/DC/SC+EEOUZd3DYqfSkMhY2/wfSeaHV1Ew4vk8e+x8lja31IbyuUa2uQ3JONqKbysw==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/parser': 7.26.2 - '@babel/types': 7.26.0 - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - jsesc: 3.0.2 - dev: true - /@babel/helper-compilation-targets@7.25.9: + '@babel/helper-compilation-targets@7.25.9': resolution: {integrity: sha512-j9Db8Suy6yV/VHa4qzrj9yZfZxhLWQdVnRlXxmKLYlhWUVB1sB2G5sxuWYXk/whHD9iW76PmNzxZ4UCnTQTVEQ==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/compat-data': 7.26.2 - '@babel/helper-validator-option': 7.25.9 - browserslist: 4.24.2 - lru-cache: 5.1.1 - semver: 6.3.1 - dev: true - /@babel/helper-module-imports@7.25.9: + '@babel/helper-module-imports@7.25.9': resolution: {integrity: sha512-tnUA4RsrmflIM6W6RFTLFSXITtl0wKjgpnLgXyowocVPrbYrLUXSBXDgTs8BlbmIzIdlBySRQjINYs2BAkiLtw==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/traverse': 7.25.9 - '@babel/types': 7.26.0 - transitivePeerDependencies: - - supports-color - dev: true - /@babel/helper-module-transforms@7.26.0(@babel/core@7.26.0): + '@babel/helper-module-transforms@7.26.0': resolution: {integrity: sha512-xO+xu6B5K2czEnQye6BHA7DolFFmS3LB7stHZFaOLb1pAwO1HWLS8fXA+eh0A2yIvltPVmx3eNNDBJA2SLHXFw==} engines: {node: '>=6.9.0'} peerDependencies: '@babel/core': ^7.0.0 - dependencies: - '@babel/core': 7.26.0 - '@babel/helper-module-imports': 7.25.9 - '@babel/helper-validator-identifier': 7.25.9 - '@babel/traverse': 7.25.9 - transitivePeerDependencies: - - supports-color - dev: true - /@babel/helper-string-parser@7.25.9: + '@babel/helper-string-parser@7.25.9': resolution: {integrity: sha512-4A/SCr/2KLd5jrtOMFzaKjVtAei3+2r/NChoBNoZ3EyP/+GlhoaEGoWOZUmFmoITP7zOJyHIMm+DYRd8o3PvHA==} engines: {node: '>=6.9.0'} - dev: true - /@babel/helper-validator-identifier@7.25.9: + '@babel/helper-validator-identifier@7.25.9': resolution: {integrity: sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==} engines: {node: '>=6.9.0'} - dev: true - /@babel/helper-validator-option@7.25.9: + '@babel/helper-validator-option@7.25.9': resolution: {integrity: sha512-e/zv1co8pp55dNdEcCynfj9X7nyUKUXoUEwfXqaZt0omVOmDe9oOTdKStH4GmAw6zxMFs50ZayuMfHDKlO7Tfw==} engines: {node: '>=6.9.0'} - dev: true - /@babel/helpers@7.26.0: + '@babel/helpers@7.26.0': resolution: {integrity: sha512-tbhNuIxNcVb21pInl3ZSjksLCvgdZy9KwJ8brv993QtIVKJBBkYXz4q4ZbAv31GdnC+R90np23L5FbEBlthAEw==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/template': 7.25.9 - '@babel/types': 7.26.0 - dev: true - /@babel/parser@7.26.2: + '@babel/parser@7.26.2': resolution: {integrity: sha512-DWMCZH9WA4Maitz2q21SRKHo9QXZxkDsbNZoVD62gusNtNBBqDg9i7uOhASfTfIGNzW+O+r7+jAlM8dwphcJKQ==} engines: {node: '>=6.0.0'} hasBin: true - dependencies: - '@babel/types': 7.26.0 - dev: true - /@babel/runtime@7.26.0: + '@babel/runtime@7.26.0': resolution: {integrity: sha512-FDSOghenHTiToteC/QRlv2q3DhPZ/oOXTBoirfWNx1Cx3TMVcGWQtMMmQcSvb/JjpNeGzx8Pq/b4fKEJuWm1sw==} engines: {node: '>=6.9.0'} - requiresBuild: true - dependencies: - regenerator-runtime: 0.14.1 - dev: true - optional: true - /@babel/template@7.25.9: + '@babel/template@7.25.9': resolution: {integrity: sha512-9DGttpmPvIxBb/2uwpVo3dqJ+O6RooAFOS+lB+xDqoE2PVCE8nfoHMdZLpfCQRLwvohzXISPZcgxt80xLfsuwg==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/code-frame': 7.26.2 - '@babel/parser': 7.26.2 - '@babel/types': 7.26.0 - dev: true - /@babel/traverse@7.25.9: + '@babel/traverse@7.25.9': resolution: {integrity: sha512-ZCuvfwOwlz/bawvAuvcj8rrithP2/N55Tzz342AkTvq4qaWbGfmCk/tKhNaV2cthijKrPAA8SRJV5WWe7IBMJw==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/code-frame': 7.26.2 - '@babel/generator': 7.26.2 - '@babel/parser': 7.26.2 - '@babel/template': 7.25.9 - '@babel/types': 7.26.0 - debug: 4.3.7 - globals: 11.12.0 - transitivePeerDependencies: - - supports-color - dev: true - /@babel/types@7.26.0: + '@babel/types@7.26.0': resolution: {integrity: sha512-Z/yiTPj+lDVnF7lWeKCIJzaIkI0vYO87dMpZ4bg4TDrFe4XXLFWL1TbXU27gBP3QccxV9mZICCrnjnYlJjXHOA==} engines: {node: '>=6.9.0'} - dependencies: - '@babel/helper-string-parser': 7.25.9 - '@babel/helper-validator-identifier': 7.25.9 - dev: true - /@effect/schema@0.75.5(effect@3.10.8): + '@changesets/apply-release-plan@7.0.5': + resolution: {integrity: sha512-1cWCk+ZshEkSVEZrm2fSj1Gz8sYvxgUL4Q78+1ZZqeqfuevPTPk033/yUZ3df8BKMohkqqHfzj0HOOrG0KtXTw==} + + '@changesets/assemble-release-plan@6.0.4': + resolution: {integrity: sha512-nqICnvmrwWj4w2x0fOhVj2QEGdlUuwVAwESrUo5HLzWMI1rE5SWfsr9ln+rDqWB6RQ2ZyaMZHUcU7/IRaUJS+Q==} + + '@changesets/changelog-git@0.2.0': + resolution: {integrity: sha512-bHOx97iFI4OClIT35Lok3sJAwM31VbUM++gnMBV16fdbtBhgYu4dxsphBF/0AZZsyAHMrnM0yFcj5gZM1py6uQ==} + + '@changesets/cli@2.27.9': + resolution: {integrity: sha512-q42a/ZbDnxPpCb5Wkm6tMVIxgeI9C/bexntzTeCFBrQEdpisQqk8kCHllYZMDjYtEc1ZzumbMJAG8H0Z4rdvjg==} + hasBin: true + + '@changesets/config@3.0.3': + resolution: {integrity: sha512-vqgQZMyIcuIpw9nqFIpTSNyc/wgm/Lu1zKN5vECy74u95Qx/Wa9g27HdgO4NkVAaq+BGA8wUc/qvbvVNs93n6A==} + + '@changesets/errors@0.2.0': + resolution: {integrity: sha512-6BLOQUscTpZeGljvyQXlWOItQyU71kCdGz7Pi8H8zdw6BI0g3m43iL4xKUVPWtG+qrrL9DTjpdn8eYuCQSRpow==} + + '@changesets/get-dependents-graph@2.1.2': + resolution: {integrity: sha512-sgcHRkiBY9i4zWYBwlVyAjEM9sAzs4wYVwJUdnbDLnVG3QwAaia1Mk5P8M7kraTOZN+vBET7n8KyB0YXCbFRLQ==} + + '@changesets/get-release-plan@4.0.4': + resolution: {integrity: sha512-SicG/S67JmPTrdcc9Vpu0wSQt7IiuN0dc8iR5VScnnTVPfIaLvKmEGRvIaF0kcn8u5ZqLbormZNTO77bCEvyWw==} + + '@changesets/get-version-range-type@0.4.0': + resolution: {integrity: sha512-hwawtob9DryoGTpixy1D3ZXbGgJu1Rhr+ySH2PvTLHvkZuQ7sRT4oQwMh0hbqZH1weAooedEjRsbrWcGLCeyVQ==} + + '@changesets/git@3.0.1': + resolution: {integrity: sha512-pdgHcYBLCPcLd82aRcuO0kxCDbw/yISlOtkmwmE8Odo1L6hSiZrBOsRl84eYG7DRCab/iHnOkWqExqc4wxk2LQ==} + + '@changesets/logger@0.1.1': + resolution: {integrity: sha512-OQtR36ZlnuTxKqoW4Sv6x5YIhOmClRd5pWsjZsddYxpWs517R0HkyiefQPIytCVh4ZcC5x9XaG8KTdd5iRQUfg==} + + '@changesets/parse@0.4.0': + resolution: {integrity: sha512-TS/9KG2CdGXS27S+QxbZXgr8uPsP4yNJYb4BC2/NeFUj80Rni3TeD2qwWmabymxmrLo7JEsytXH1FbpKTbvivw==} + + '@changesets/pre@2.0.1': + resolution: {integrity: sha512-vvBJ/If4jKM4tPz9JdY2kGOgWmCowUYOi5Ycv8dyLnEE8FgpYYUo1mgJZxcdtGGP3aG8rAQulGLyyXGSLkIMTQ==} + + '@changesets/read@0.6.1': + resolution: {integrity: sha512-jYMbyXQk3nwP25nRzQQGa1nKLY0KfoOV7VLgwucI0bUO8t8ZLCr6LZmgjXsiKuRDc+5A6doKPr9w2d+FEJ55zQ==} + + '@changesets/should-skip-package@0.1.1': + resolution: {integrity: sha512-H9LjLbF6mMHLtJIc/eHR9Na+MifJ3VxtgP/Y+XLn4BF7tDTEN1HNYtH6QMcjP1uxp9sjaFYmW8xqloaCi/ckTg==} + + '@changesets/types@4.1.0': + resolution: {integrity: sha512-LDQvVDv5Kb50ny2s25Fhm3d9QSZimsoUGBsUioj6MC3qbMUCuC8GPIvk/M6IvXx3lYhAs0lwWUQLb+VIEUCECw==} + + '@changesets/types@6.0.0': + resolution: {integrity: sha512-b1UkfNulgKoWfqyHtzKS5fOZYSJO+77adgL7DLRDr+/7jhChN+QcHnbjiQVOz/U+Ts3PGNySq7diAItzDgugfQ==} + + '@changesets/write@0.3.2': + resolution: {integrity: sha512-kDxDrPNpUgsjDbWBvUo27PzKX4gqeKOlhibaOXDJA6kuBisGqNHv/HwGJrAu8U/dSf8ZEFIeHIPtvSlZI1kULw==} + + '@dependents/detective-less@5.0.0': + resolution: {integrity: sha512-D/9dozteKcutI5OdxJd8rU+fL6XgaaRg60sPPJWkT33OCiRfkCu5wO5B/yXTaaL2e6EB0lcCBGe5E0XscZCvvQ==} + engines: {node: '>=18'} + + '@effect/schema@0.75.5': resolution: {integrity: sha512-TQInulTVCuF+9EIbJpyLP6dvxbQJMphrnRqgexm/Ze39rSjfhJuufF7XvU3SxTgg3HnL7B/kpORTJbHhlE6thw==} - requiresBuild: true peerDependencies: effect: ^3.9.2 - dependencies: - effect: 3.10.8 - fast-check: 3.23.0 - dev: true - optional: true - /@esbuild/aix-ppc64@0.21.5: + '@esbuild/aix-ppc64@0.21.5': resolution: {integrity: sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==} engines: {node: '>=12'} cpu: [ppc64] os: [aix] - requiresBuild: true - dev: true - optional: true - /@esbuild/android-arm64@0.21.5: + '@esbuild/aix-ppc64@0.24.0': + resolution: {integrity: sha512-WtKdFM7ls47zkKHFVzMz8opM7LkcsIp9amDUBIAWirg70RM71WRSjdILPsY5Uv1D42ZpUfaPILDlfactHgsRkw==} + engines: {node: '>=18'} + cpu: [ppc64] + os: [aix] + + '@esbuild/android-arm64@0.21.5': resolution: {integrity: sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==} engines: {node: '>=12'} cpu: [arm64] os: [android] - requiresBuild: true - dev: true - optional: true - /@esbuild/android-arm@0.21.5: + '@esbuild/android-arm64@0.24.0': + resolution: {integrity: sha512-Vsm497xFM7tTIPYK9bNTYJyF/lsP590Qc1WxJdlB6ljCbdZKU9SY8i7+Iin4kyhV/KV5J2rOKsBQbB77Ab7L/w==} + engines: {node: '>=18'} + cpu: [arm64] + os: [android] + + '@esbuild/android-arm@0.21.5': resolution: {integrity: sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==} engines: {node: '>=12'} cpu: [arm] os: [android] - requiresBuild: true - dev: true - optional: true - /@esbuild/android-x64@0.21.5: + '@esbuild/android-arm@0.24.0': + resolution: {integrity: sha512-arAtTPo76fJ/ICkXWetLCc9EwEHKaeya4vMrReVlEIUCAUncH7M4bhMQ+M9Vf+FFOZJdTNMXNBrWwW+OXWpSew==} + engines: {node: '>=18'} + cpu: [arm] + os: [android] + + '@esbuild/android-x64@0.21.5': resolution: {integrity: sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==} engines: {node: '>=12'} cpu: [x64] os: [android] - requiresBuild: true - dev: true - optional: true - /@esbuild/darwin-arm64@0.21.5: + '@esbuild/android-x64@0.24.0': + resolution: {integrity: sha512-t8GrvnFkiIY7pa7mMgJd7p8p8qqYIz1NYiAoKc75Zyv73L3DZW++oYMSHPRarcotTKuSs6m3hTOa5CKHaS02TQ==} + engines: {node: '>=18'} + cpu: [x64] + os: [android] + + '@esbuild/darwin-arm64@0.21.5': resolution: {integrity: sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==} engines: {node: '>=12'} cpu: [arm64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /@esbuild/darwin-x64@0.21.5: + '@esbuild/darwin-arm64@0.24.0': + resolution: {integrity: sha512-CKyDpRbK1hXwv79soeTJNHb5EiG6ct3efd/FTPdzOWdbZZfGhpbcqIpiD0+vwmpu0wTIL97ZRPZu8vUt46nBSw==} + engines: {node: '>=18'} + cpu: [arm64] + os: [darwin] + + '@esbuild/darwin-x64@0.21.5': resolution: {integrity: sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==} engines: {node: '>=12'} cpu: [x64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /@esbuild/freebsd-arm64@0.21.5: + '@esbuild/darwin-x64@0.24.0': + resolution: {integrity: sha512-rgtz6flkVkh58od4PwTRqxbKH9cOjaXCMZgWD905JOzjFKW+7EiUObfd/Kav+A6Gyud6WZk9w+xu6QLytdi2OA==} + engines: {node: '>=18'} + cpu: [x64] + os: [darwin] + + '@esbuild/freebsd-arm64@0.21.5': resolution: {integrity: sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==} engines: {node: '>=12'} cpu: [arm64] os: [freebsd] - requiresBuild: true - dev: true - optional: true - /@esbuild/freebsd-x64@0.21.5: + '@esbuild/freebsd-arm64@0.24.0': + resolution: {integrity: sha512-6Mtdq5nHggwfDNLAHkPlyLBpE5L6hwsuXZX8XNmHno9JuL2+bg2BX5tRkwjyfn6sKbxZTq68suOjgWqCicvPXA==} + engines: {node: '>=18'} + cpu: [arm64] + os: [freebsd] + + '@esbuild/freebsd-x64@0.21.5': resolution: {integrity: sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==} engines: {node: '>=12'} cpu: [x64] os: [freebsd] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-arm64@0.21.5: + '@esbuild/freebsd-x64@0.24.0': + resolution: {integrity: sha512-D3H+xh3/zphoX8ck4S2RxKR6gHlHDXXzOf6f/9dbFt/NRBDIE33+cVa49Kil4WUjxMGW0ZIYBYtaGCa2+OsQwQ==} + engines: {node: '>=18'} + cpu: [x64] + os: [freebsd] + + '@esbuild/linux-arm64@0.21.5': resolution: {integrity: sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==} engines: {node: '>=12'} cpu: [arm64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-arm@0.21.5: + '@esbuild/linux-arm64@0.24.0': + resolution: {integrity: sha512-TDijPXTOeE3eaMkRYpcy3LarIg13dS9wWHRdwYRnzlwlA370rNdZqbcp0WTyyV/k2zSxfko52+C7jU5F9Tfj1g==} + engines: {node: '>=18'} + cpu: [arm64] + os: [linux] + + '@esbuild/linux-arm@0.21.5': resolution: {integrity: sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==} engines: {node: '>=12'} cpu: [arm] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-ia32@0.21.5: + '@esbuild/linux-arm@0.24.0': + resolution: {integrity: sha512-gJKIi2IjRo5G6Glxb8d3DzYXlxdEj2NlkixPsqePSZMhLudqPhtZ4BUrpIuTjJYXxvF9njql+vRjB2oaC9XpBw==} + engines: {node: '>=18'} + cpu: [arm] + os: [linux] + + '@esbuild/linux-ia32@0.21.5': resolution: {integrity: sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==} engines: {node: '>=12'} cpu: [ia32] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-loong64@0.21.5: + '@esbuild/linux-ia32@0.24.0': + resolution: {integrity: sha512-K40ip1LAcA0byL05TbCQ4yJ4swvnbzHscRmUilrmP9Am7//0UjPreh4lpYzvThT2Quw66MhjG//20mrufm40mA==} + engines: {node: '>=18'} + cpu: [ia32] + os: [linux] + + '@esbuild/linux-loong64@0.21.5': resolution: {integrity: sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==} engines: {node: '>=12'} cpu: [loong64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-mips64el@0.21.5: + '@esbuild/linux-loong64@0.24.0': + resolution: {integrity: sha512-0mswrYP/9ai+CU0BzBfPMZ8RVm3RGAN/lmOMgW4aFUSOQBjA31UP8Mr6DDhWSuMwj7jaWOT0p0WoZ6jeHhrD7g==} + engines: {node: '>=18'} + cpu: [loong64] + os: [linux] + + '@esbuild/linux-mips64el@0.21.5': resolution: {integrity: sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==} engines: {node: '>=12'} cpu: [mips64el] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-ppc64@0.21.5: + '@esbuild/linux-mips64el@0.24.0': + resolution: {integrity: sha512-hIKvXm0/3w/5+RDtCJeXqMZGkI2s4oMUGj3/jM0QzhgIASWrGO5/RlzAzm5nNh/awHE0A19h/CvHQe6FaBNrRA==} + engines: {node: '>=18'} + cpu: [mips64el] + os: [linux] + + '@esbuild/linux-ppc64@0.21.5': resolution: {integrity: sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==} engines: {node: '>=12'} cpu: [ppc64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-riscv64@0.21.5: + '@esbuild/linux-ppc64@0.24.0': + resolution: {integrity: sha512-HcZh5BNq0aC52UoocJxaKORfFODWXZxtBaaZNuN3PUX3MoDsChsZqopzi5UupRhPHSEHotoiptqikjN/B77mYQ==} + engines: {node: '>=18'} + cpu: [ppc64] + os: [linux] + + '@esbuild/linux-riscv64@0.21.5': resolution: {integrity: sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==} engines: {node: '>=12'} cpu: [riscv64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-s390x@0.21.5: + '@esbuild/linux-riscv64@0.24.0': + resolution: {integrity: sha512-bEh7dMn/h3QxeR2KTy1DUszQjUrIHPZKyO6aN1X4BCnhfYhuQqedHaa5MxSQA/06j3GpiIlFGSsy1c7Gf9padw==} + engines: {node: '>=18'} + cpu: [riscv64] + os: [linux] + + '@esbuild/linux-s390x@0.21.5': resolution: {integrity: sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==} engines: {node: '>=12'} cpu: [s390x] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/linux-x64@0.21.5: + '@esbuild/linux-s390x@0.24.0': + resolution: {integrity: sha512-ZcQ6+qRkw1UcZGPyrCiHHkmBaj9SiCD8Oqd556HldP+QlpUIe2Wgn3ehQGVoPOvZvtHm8HPx+bH20c9pvbkX3g==} + engines: {node: '>=18'} + cpu: [s390x] + os: [linux] + + '@esbuild/linux-x64@0.21.5': resolution: {integrity: sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==} engines: {node: '>=12'} cpu: [x64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@esbuild/netbsd-x64@0.21.5: + '@esbuild/linux-x64@0.24.0': + resolution: {integrity: sha512-vbutsFqQ+foy3wSSbmjBXXIJ6PL3scghJoM8zCL142cGaZKAdCZHyf+Bpu/MmX9zT9Q0zFBVKb36Ma5Fzfa8xA==} + engines: {node: '>=18'} + cpu: [x64] + os: [linux] + + '@esbuild/netbsd-x64@0.21.5': resolution: {integrity: sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==} engines: {node: '>=12'} cpu: [x64] os: [netbsd] - requiresBuild: true - dev: true - optional: true - /@esbuild/openbsd-x64@0.21.5: + '@esbuild/netbsd-x64@0.24.0': + resolution: {integrity: sha512-hjQ0R/ulkO8fCYFsG0FZoH+pWgTTDreqpqY7UnQntnaKv95uP5iW3+dChxnx7C3trQQU40S+OgWhUVwCjVFLvg==} + engines: {node: '>=18'} + cpu: [x64] + os: [netbsd] + + '@esbuild/openbsd-arm64@0.24.0': + resolution: {integrity: sha512-MD9uzzkPQbYehwcN583yx3Tu5M8EIoTD+tUgKF982WYL9Pf5rKy9ltgD0eUgs8pvKnmizxjXZyLt0z6DC3rRXg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [openbsd] + + '@esbuild/openbsd-x64@0.21.5': resolution: {integrity: sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==} engines: {node: '>=12'} cpu: [x64] os: [openbsd] - requiresBuild: true - dev: true - optional: true - /@esbuild/sunos-x64@0.21.5: + '@esbuild/openbsd-x64@0.24.0': + resolution: {integrity: sha512-4ir0aY1NGUhIC1hdoCzr1+5b43mw99uNwVzhIq1OY3QcEwPDO3B7WNXBzaKY5Nsf1+N11i1eOfFcq+D/gOS15Q==} + engines: {node: '>=18'} + cpu: [x64] + os: [openbsd] + + '@esbuild/sunos-x64@0.21.5': resolution: {integrity: sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==} engines: {node: '>=12'} cpu: [x64] os: [sunos] - requiresBuild: true - dev: true - optional: true - /@esbuild/win32-arm64@0.21.5: + '@esbuild/sunos-x64@0.24.0': + resolution: {integrity: sha512-jVzdzsbM5xrotH+W5f1s+JtUy1UWgjU0Cf4wMvffTB8m6wP5/kx0KiaLHlbJO+dMgtxKV8RQ/JvtlFcdZ1zCPA==} + engines: {node: '>=18'} + cpu: [x64] + os: [sunos] + + '@esbuild/win32-arm64@0.21.5': resolution: {integrity: sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==} engines: {node: '>=12'} cpu: [arm64] os: [win32] - requiresBuild: true - dev: true - optional: true - /@esbuild/win32-ia32@0.21.5: + '@esbuild/win32-arm64@0.24.0': + resolution: {integrity: sha512-iKc8GAslzRpBytO2/aN3d2yb2z8XTVfNV0PjGlCxKo5SgWmNXx82I/Q3aG1tFfS+A2igVCY97TJ8tnYwpUWLCA==} + engines: {node: '>=18'} + cpu: [arm64] + os: [win32] + + '@esbuild/win32-ia32@0.21.5': resolution: {integrity: sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==} engines: {node: '>=12'} cpu: [ia32] os: [win32] - requiresBuild: true - dev: true - optional: true - /@esbuild/win32-x64@0.21.5: + '@esbuild/win32-ia32@0.24.0': + resolution: {integrity: sha512-vQW36KZolfIudCcTnaTpmLQ24Ha1RjygBo39/aLkM2kmjkWmZGEJ5Gn9l5/7tzXA42QGIoWbICfg6KLLkIw6yw==} + engines: {node: '>=18'} + cpu: [ia32] + os: [win32] + + '@esbuild/win32-x64@0.21.5': resolution: {integrity: sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==} engines: {node: '>=12'} cpu: [x64] os: [win32] - requiresBuild: true - dev: true - optional: true - /@eslint-community/eslint-utils@4.4.1(eslint@8.57.1): + '@esbuild/win32-x64@0.24.0': + resolution: {integrity: sha512-7IAFPrjSQIJrGsK6flwg7NFmwBoSTyF3rl7If0hNUFQU4ilTsEPL6GuMuU9BfIWVVGuRnuIidkSMC+c0Otu8IA==} + engines: {node: '>=18'} + cpu: [x64] + os: [win32] + + '@eslint-community/eslint-utils@4.4.1': resolution: {integrity: sha512-s3O3waFUrMV8P/XaF/+ZTp1X9XBZW1a4B97ZnjQF2KYWaFD2A8KyFBsrsfSjEmjn3RGWAIuvlneuZm3CUK3jbA==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} peerDependencies: eslint: ^6.0.0 || ^7.0.0 || >=8.0.0 - dependencies: - eslint: 8.57.1 - eslint-visitor-keys: 3.4.3 - dev: false - /@eslint-community/regexpp@4.12.1: + '@eslint-community/regexpp@4.12.1': resolution: {integrity: sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==} engines: {node: ^12.0.0 || ^14.0.0 || >=16.0.0} - dev: false - /@eslint/eslintrc@2.1.4: + '@eslint/eslintrc@2.1.4': resolution: {integrity: sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - dependencies: - ajv: 6.12.6 - debug: 4.3.7 - espree: 9.6.1 - globals: 13.24.0 - ignore: 5.3.2 - import-fresh: 3.3.0 - js-yaml: 4.1.0 - minimatch: 3.1.2 - strip-json-comments: 3.1.1 - transitivePeerDependencies: - - supports-color - dev: false - /@eslint/js@8.57.1: + '@eslint/js@8.57.1': resolution: {integrity: sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - dev: false - /@exodus/schemasafe@1.3.0: + '@exodus/schemasafe@1.3.0': resolution: {integrity: sha512-5Aap/GaRupgNx/feGBwLLTVv8OQFfv3pq2lPRzPg9R+IOBnDgghTGW7l7EuVXOvg5cc/xSAlRW8rBrjIC3Nvqw==} - requiresBuild: true - dev: true - optional: true - /@floating-ui/core@1.6.8: + '@floating-ui/core@1.6.8': resolution: {integrity: sha512-7XJ9cPU+yI2QeLS+FCSlqNFZJq8arvswefkZrYI1yQBbftw6FyrZOxYSh+9S7z7TpeWlRt9zJ5IhM1WIL334jA==} - dependencies: - '@floating-ui/utils': 0.2.8 - dev: true - /@floating-ui/dom@1.6.12: + '@floating-ui/dom@1.6.12': resolution: {integrity: sha512-NP83c0HjokcGVEMeoStg317VD9W7eDlGK7457dMBANbKA6GJZdc7rjujdgqzTaz93jkGgc5P/jeWbaCHnMNc+w==} - dependencies: - '@floating-ui/core': 1.6.8 - '@floating-ui/utils': 0.2.8 - dev: true - /@floating-ui/utils@0.2.8: + '@floating-ui/utils@0.2.8': resolution: {integrity: sha512-kym7SodPp8/wloecOpcmSnWJsK7M0E5Wg8UcFA+uO4B9s5d0ywXOEro/8HM9x0rW+TljRzul/14UYz3TleT3ig==} - dev: true - /@gcornut/valibot-json-schema@0.31.0: + '@gcornut/valibot-json-schema@0.31.0': resolution: {integrity: sha512-3xGptCurm23e7nuPQkdrE5rEs1FeTPHhAUsBuwwqG4/YeZLwJOoYZv+fmsppUEfo5y9lzUwNQrNqLS/q7HMc7g==} hasBin: true - requiresBuild: true - dependencies: - valibot: 0.31.1 - optionalDependencies: - '@types/json-schema': 7.0.15 - esbuild: 0.21.5 - esbuild-runner: 2.2.2(esbuild@0.21.5) - dev: true - optional: true - /@hapi/hoek@9.3.0: + '@gcornut/valibot-json-schema@0.42.0': + resolution: {integrity: sha512-4Et4AN6wmqeA0PfU5Clkv/IS27wiefsWf6TemAZrb75uzkClYEFavim7SboeKwbll9Nbsn2Iv0LT/HS5H7orZg==} + hasBin: true + + '@hapi/hoek@9.3.0': resolution: {integrity: sha512-/c6rf4UJlmHlC9b5BaNvzAcFv7HZ2QHaV0D4/HNlBdvFnvQq8RI4kYdhyPCl7Xj+oWvTWQ8ujhqS53LIgAe6KQ==} - requiresBuild: true - dev: true - optional: true - /@hapi/topo@5.1.0: + '@hapi/topo@5.1.0': resolution: {integrity: sha512-foQZKJig7Ob0BMAYBfcJk8d77QtOe7Wo4ox7ff1lQYoNNAb6jwcY1ncdoy2e9wQZzvNy7ODZCYJkK8kzmcAnAg==} - requiresBuild: true - dependencies: - '@hapi/hoek': 9.3.0 - dev: true - optional: true - /@humanwhocodes/config-array@0.13.0: + '@huakunshen/comlink@4.4.1': + resolution: {integrity: sha512-rxWFQ0NEc28z/r6Ixy0ZllNaRqvkqbmuT3oTwzVnwVyDFdIkqoicrwr1Z718ZQbWoVFBdmqSjm93N273ludHMA==} + + '@humanwhocodes/config-array@0.13.0': resolution: {integrity: sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==} engines: {node: '>=10.10.0'} deprecated: Use @eslint/config-array instead - dependencies: - '@humanwhocodes/object-schema': 2.0.3 - debug: 4.3.7 - minimatch: 3.1.2 - transitivePeerDependencies: - - supports-color - dev: false - /@humanwhocodes/module-importer@1.0.1: + '@humanwhocodes/module-importer@1.0.1': resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==} engines: {node: '>=12.22'} - dev: false - /@humanwhocodes/object-schema@2.0.3: + '@humanwhocodes/object-schema@2.0.3': resolution: {integrity: sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==} deprecated: Use @eslint/object-schema instead - dev: false - /@ianvs/prettier-plugin-sort-imports@4.3.1(prettier@3.3.3): + '@ianvs/prettier-plugin-sort-imports@4.3.1': resolution: {integrity: sha512-ZHwbyjkANZOjaBm3ZosADD2OUYGFzQGxfy67HmGZU94mHqe7g1LCMA7YYKB1Cq+UTPCBqlAYapY0KXAjKEw8Sg==} peerDependencies: '@vue/compiler-sfc': 2.7.x || 3.x @@ -689,574 +1314,710 @@ packages: peerDependenciesMeta: '@vue/compiler-sfc': optional: true - dependencies: - '@babel/core': 7.26.0 - '@babel/generator': 7.26.2 - '@babel/parser': 7.26.2 - '@babel/traverse': 7.25.9 - '@babel/types': 7.26.0 - prettier: 3.3.3 - semver: 7.6.3 - transitivePeerDependencies: - - supports-color - dev: true - /@internationalized/date@3.5.6: + '@iconify/svelte@4.0.2': + resolution: {integrity: sha512-6BSrU85FzGfhQD3bTXpnkCCvBAglEt8T9QednVnXAYm4C+d3464y+pYMzhQNJm5mPId2cuiw+2wXlDflXllHDw==} + peerDependencies: + svelte: '>4.0.0' + + '@iconify/types@2.0.0': + resolution: {integrity: sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg==} + + '@internationalized/date@3.5.6': resolution: {integrity: sha512-jLxQjefH9VI5P9UQuqB6qNKnvFt1Ky1TPIzHGsIlCi7sZZoMR8SdYbBGRvM0y+Jtb+ez4ieBzmiAUcpmPYpyOw==} - dependencies: - '@swc/helpers': 0.5.13 - dev: true - /@isaacs/cliui@8.0.2: + '@isaacs/cliui@8.0.2': resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} engines: {node: '>=12'} - dependencies: - string-width: 5.1.2 - string-width-cjs: /string-width@4.2.3 - strip-ansi: 7.1.0 - strip-ansi-cjs: /strip-ansi@6.0.1 - wrap-ansi: 8.1.0 - wrap-ansi-cjs: /wrap-ansi@7.0.0 - dev: true - /@jridgewell/gen-mapping@0.3.5: + '@isaacs/fs-minipass@4.0.1': + resolution: {integrity: sha512-wgm9Ehl2jpeqP3zw/7mo3kRHFp5MEDhqAdwy1fTGkHAwnkGOVsgpvQhL8B5n1qlb01jV3n/bI0ZfZp5lWA1k4w==} + engines: {node: '>=18.0.0'} + + '@jridgewell/gen-mapping@0.3.5': resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==} engines: {node: '>=6.0.0'} - dependencies: - '@jridgewell/set-array': 1.2.1 - '@jridgewell/sourcemap-codec': 1.5.0 - '@jridgewell/trace-mapping': 0.3.25 - /@jridgewell/resolve-uri@3.1.2: + '@jridgewell/resolve-uri@3.1.2': resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} engines: {node: '>=6.0.0'} - /@jridgewell/set-array@1.2.1: + '@jridgewell/set-array@1.2.1': resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==} engines: {node: '>=6.0.0'} - /@jridgewell/sourcemap-codec@1.5.0: + '@jridgewell/source-map@0.3.6': + resolution: {integrity: sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ==} + + '@jridgewell/sourcemap-codec@1.5.0': resolution: {integrity: sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==} - /@jridgewell/trace-mapping@0.3.25: + '@jridgewell/trace-mapping@0.3.25': resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==} - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.5.0 - /@kksh/svelte5@0.1.2-beta.3(lucide-svelte@0.416.0)(svelte-sonner@0.3.28)(svelte@5.1.9): - resolution: {integrity: sha512-N0QvlzpHvncMW1SSyzRqVhhjyTHN6PXT8WciVek0/3GfZOU8qBEbcywL27LX8LofqbGwXv5AnLCO2vJifu/fgQ==} + '@jsr/std__semver@1.0.3': + resolution: {integrity: sha512-d1uBT0Muxhd3yBIw9ZE1Q/4N1Y0td0EJe1AqwM3hP05IMwaWQV/miksQOPR3rup3bVovuIvqBm7WJcoUripdQA==, tarball: https://npm.jsr.io/~/11/@jsr/std__semver/1.0.3.tgz} + + '@kksh/svelte5@0.1.2-beta.4': + resolution: {integrity: sha512-QUA3wl4aOUcnfTo37/l37fWlhwf0lbtnNugDiwBJgrjaLeNF6nEzlYFKxxsA7erR6hY6VZCeVlf01S3ZgPva4w==} peerDependencies: lucide-svelte: ^0.416.0 svelte: ^5.0.0 svelte-sonner: ^0.3.27 - dependencies: - lucide-svelte: 0.416.0(svelte@5.1.9) - svelte: 5.1.9 - svelte-persisted-store: 0.11.0(svelte@5.1.9) - svelte-sonner: 0.3.28(svelte@5.1.9) - dev: true - /@melt-ui/svelte@0.76.2(svelte@5.1.9): + '@manypkg/find-root@1.1.0': + resolution: {integrity: sha512-mki5uBvhHzO8kYYix/WRy2WX8S3B5wdVSc9D6KcU5lQNglP2yt58/VfLuAK49glRXChosY8ap2oJ1qgma3GUVA==} + + '@manypkg/get-packages@1.1.3': + resolution: {integrity: sha512-fo+QhuU3qE/2TQMQmbVMqaQ6EWbMhi4ABWP+O4AM1NqPBuy0OrApV5LO6BrrgnhtAHS2NH6RrVk9OL181tTi8A==} + + '@melt-ui/svelte@0.76.2': resolution: {integrity: sha512-7SbOa11tXUS95T3fReL+dwDs5FyJtCEqrqG3inRziDws346SYLsxOQ6HmX+4BkIsQh1R8U3XNa+EMmdMt38lMA==} peerDependencies: svelte: '>=3 <5' - dependencies: - '@floating-ui/core': 1.6.8 - '@floating-ui/dom': 1.6.12 - '@internationalized/date': 3.5.6 - dequal: 2.0.3 - focus-trap: 7.6.0 - nanoid: 5.0.8 - svelte: 5.1.9 - dev: true - /@nodelib/fs.scandir@2.1.5: + '@nodelib/fs.scandir@2.1.5': resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} engines: {node: '>= 8'} - dependencies: - '@nodelib/fs.stat': 2.0.5 - run-parallel: 1.2.0 - /@nodelib/fs.stat@2.0.5: + '@nodelib/fs.stat@2.0.5': resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} engines: {node: '>= 8'} - /@nodelib/fs.walk@1.2.8: + '@nodelib/fs.walk@1.2.8': resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} engines: {node: '>= 8'} - dependencies: - '@nodelib/fs.scandir': 2.1.5 - fastq: 1.17.1 - /@pkgjs/parseargs@0.11.0: + '@pkgjs/parseargs@0.11.0': resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} engines: {node: '>=14'} - requiresBuild: true - dev: true - optional: true - /@polka/url@1.0.0-next.28: + '@polka/url@1.0.0-next.28': resolution: {integrity: sha512-8LduaNlMZGwdZ6qWrKlfa+2M4gahzFkprZiAt2TF8uS0qQgBizKXpXURqvTJ4WtmupWxaLqjRb2UCTe72mu+Aw==} - dev: true - /@poppinss/macroable@1.0.3: + '@poppinss/macroable@1.0.3': resolution: {integrity: sha512-B4iV6QxW//Fn17+qF1EMZRmoThIUJlCtcO85yoRDJnMyHeAthjz4ig9OTkfGGXKtQhcdPX0me75gU5K9J897+w==} engines: {node: '>=18.16.0'} - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-android-arm-eabi@4.24.3: + '@rollup/plugin-alias@5.1.1': + resolution: {integrity: sha512-PR9zDb+rOzkRb2VD+EuKB7UC41vU5DIwZ5qqCpk0KJudcWAyi8rvYOhS7+L5aZCspw1stTViLgN5v6FF1p5cgQ==} + engines: {node: '>=14.0.0'} + peerDependencies: + rollup: ^1.20.0||^2.0.0||^3.0.0||^4.0.0 + peerDependenciesMeta: + rollup: + optional: true + + '@rollup/plugin-typescript@11.1.6': + resolution: {integrity: sha512-R92yOmIACgYdJ7dJ97p4K69I8gg6IEHt8M7dUBxN3W6nrO8uUxX5ixl0yU/N3aZTi8WhPuICvOHXQvF6FaykAA==} + engines: {node: '>=14.0.0'} + peerDependencies: + rollup: ^2.14.0||^3.0.0||^4.0.0 + tslib: '*' + typescript: '>=3.7.0' + peerDependenciesMeta: + rollup: + optional: true + tslib: + optional: true + + '@rollup/pluginutils@5.1.3': + resolution: {integrity: sha512-Pnsb6f32CD2W3uCaLZIzDmeFyQ2b8UWMFI7xtwUezpcGBDVDW6y9XgAWIlARiGAo6eNF5FK5aQTr0LFyNyqq5A==} + engines: {node: '>=14.0.0'} + peerDependencies: + rollup: ^1.20.0||^2.0.0||^3.0.0||^4.0.0 + peerDependenciesMeta: + rollup: + optional: true + + '@rollup/rollup-android-arm-eabi@4.24.3': resolution: {integrity: sha512-ufb2CH2KfBWPJok95frEZZ82LtDl0A6QKTa8MoM+cWwDZvVGl5/jNb79pIhRvAalUu+7LD91VYR0nwRD799HkQ==} cpu: [arm] os: [android] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-android-arm64@4.24.3: + '@rollup/rollup-android-arm64@4.24.3': resolution: {integrity: sha512-iAHpft/eQk9vkWIV5t22V77d90CRofgR2006UiCjHcHJFVI1E0oBkQIAbz+pLtthFw3hWEmVB4ilxGyBf48i2Q==} cpu: [arm64] os: [android] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-darwin-arm64@4.24.3: + '@rollup/rollup-darwin-arm64@4.24.3': resolution: {integrity: sha512-QPW2YmkWLlvqmOa2OwrfqLJqkHm7kJCIMq9kOz40Zo9Ipi40kf9ONG5Sz76zszrmIZZ4hgRIkez69YnTHgEz1w==} cpu: [arm64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-darwin-x64@4.24.3: + '@rollup/rollup-darwin-x64@4.24.3': resolution: {integrity: sha512-KO0pN5x3+uZm1ZXeIfDqwcvnQ9UEGN8JX5ufhmgH5Lz4ujjZMAnxQygZAVGemFWn+ZZC0FQopruV4lqmGMshow==} cpu: [x64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-freebsd-arm64@4.24.3: + '@rollup/rollup-freebsd-arm64@4.24.3': resolution: {integrity: sha512-CsC+ZdIiZCZbBI+aRlWpYJMSWvVssPuWqrDy/zi9YfnatKKSLFCe6fjna1grHuo/nVaHG+kiglpRhyBQYRTK4A==} cpu: [arm64] os: [freebsd] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-freebsd-x64@4.24.3: + '@rollup/rollup-freebsd-x64@4.24.3': resolution: {integrity: sha512-F0nqiLThcfKvRQhZEzMIXOQG4EeX61im61VYL1jo4eBxv4aZRmpin6crnBJQ/nWnCsjH5F6J3W6Stdm0mBNqBg==} cpu: [x64] os: [freebsd] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-arm-gnueabihf@4.24.3: + '@rollup/rollup-linux-arm-gnueabihf@4.24.3': resolution: {integrity: sha512-KRSFHyE/RdxQ1CSeOIBVIAxStFC/hnBgVcaiCkQaVC+EYDtTe4X7z5tBkFyRoBgUGtB6Xg6t9t2kulnX6wJc6A==} cpu: [arm] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-arm-musleabihf@4.24.3: + '@rollup/rollup-linux-arm-musleabihf@4.24.3': resolution: {integrity: sha512-h6Q8MT+e05zP5BxEKz0vi0DhthLdrNEnspdLzkoFqGwnmOzakEHSlXfVyA4HJ322QtFy7biUAVFPvIDEDQa6rw==} cpu: [arm] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-arm64-gnu@4.24.3: + '@rollup/rollup-linux-arm64-gnu@4.24.3': resolution: {integrity: sha512-fKElSyXhXIJ9pqiYRqisfirIo2Z5pTTve5K438URf08fsypXrEkVmShkSfM8GJ1aUyvjakT+fn2W7Czlpd/0FQ==} cpu: [arm64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-arm64-musl@4.24.3: + '@rollup/rollup-linux-arm64-musl@4.24.3': resolution: {integrity: sha512-YlddZSUk8G0px9/+V9PVilVDC6ydMz7WquxozToozSnfFK6wa6ne1ATUjUvjin09jp34p84milxlY5ikueoenw==} cpu: [arm64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-powerpc64le-gnu@4.24.3: + '@rollup/rollup-linux-powerpc64le-gnu@4.24.3': resolution: {integrity: sha512-yNaWw+GAO8JjVx3s3cMeG5Esz1cKVzz8PkTJSfYzE5u7A+NvGmbVFEHP+BikTIyYWuz0+DX9kaA3pH9Sqxp69g==} cpu: [ppc64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-riscv64-gnu@4.24.3: + '@rollup/rollup-linux-riscv64-gnu@4.24.3': resolution: {integrity: sha512-lWKNQfsbpv14ZCtM/HkjCTm4oWTKTfxPmr7iPfp3AHSqyoTz5AgLemYkWLwOBWc+XxBbrU9SCokZP0WlBZM9lA==} cpu: [riscv64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-s390x-gnu@4.24.3: + '@rollup/rollup-linux-s390x-gnu@4.24.3': resolution: {integrity: sha512-HoojGXTC2CgCcq0Woc/dn12wQUlkNyfH0I1ABK4Ni9YXyFQa86Fkt2Q0nqgLfbhkyfQ6003i3qQk9pLh/SpAYw==} cpu: [s390x] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-x64-gnu@4.24.3: + '@rollup/rollup-linux-x64-gnu@4.24.3': resolution: {integrity: sha512-mnEOh4iE4USSccBOtcrjF5nj+5/zm6NcNhbSEfR3Ot0pxBwvEn5QVUXcuOwwPkapDtGZ6pT02xLoPaNv06w7KQ==} cpu: [x64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-linux-x64-musl@4.24.3: + '@rollup/rollup-linux-x64-musl@4.24.3': resolution: {integrity: sha512-rMTzawBPimBQkG9NKpNHvquIUTQPzrnPxPbCY1Xt+mFkW7pshvyIS5kYgcf74goxXOQk0CP3EoOC1zcEezKXhw==} cpu: [x64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-win32-arm64-msvc@4.24.3: + '@rollup/rollup-win32-arm64-msvc@4.24.3': resolution: {integrity: sha512-2lg1CE305xNvnH3SyiKwPVsTVLCg4TmNCF1z7PSHX2uZY2VbUpdkgAllVoISD7JO7zu+YynpWNSKAtOrX3AiuA==} cpu: [arm64] os: [win32] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-win32-ia32-msvc@4.24.3: + '@rollup/rollup-win32-ia32-msvc@4.24.3': resolution: {integrity: sha512-9SjYp1sPyxJsPWuhOCX6F4jUMXGbVVd5obVpoVEi8ClZqo52ViZewA6eFz85y8ezuOA+uJMP5A5zo6Oz4S5rVQ==} cpu: [ia32] os: [win32] - requiresBuild: true - dev: true - optional: true - /@rollup/rollup-win32-x64-msvc@4.24.3: + '@rollup/rollup-win32-x64-msvc@4.24.3': resolution: {integrity: sha512-HGZgRFFYrMrP3TJlq58nR1xy8zHKId25vhmm5S9jETEfDf6xybPxsavFTJaufe2zgOGYJBskGlj49CwtEuFhWQ==} cpu: [x64] os: [win32] - requiresBuild: true - dev: true - optional: true - /@sideway/address@4.1.5: + '@shikijs/core@1.22.2': + resolution: {integrity: sha512-bvIQcd8BEeR1yFvOYv6HDiyta2FFVePbzeowf5pPS1avczrPK+cjmaxxh0nx5QzbON7+Sv0sQfQVciO7bN72sg==} + + '@shikijs/engine-javascript@1.22.2': + resolution: {integrity: sha512-iOvql09ql6m+3d1vtvP8fLCVCK7BQD1pJFmHIECsujB0V32BJ0Ab6hxk1ewVSMFA58FI0pR2Had9BKZdyQrxTw==} + + '@shikijs/engine-oniguruma@1.22.2': + resolution: {integrity: sha512-GIZPAGzQOy56mGvWMoZRPggn0dTlBf1gutV5TdceLCZlFNqWmuc7u+CzD0Gd9vQUTgLbrt0KLzz6FNprqYAxlA==} + + '@shikijs/types@1.22.2': + resolution: {integrity: sha512-NCWDa6LGZqTuzjsGfXOBWfjS/fDIbDdmVDug+7ykVe1IKT4c1gakrvlfFYp5NhAXH/lyqLM8wsAPo5wNy73Feg==} + + '@shikijs/vscode-textmate@9.3.0': + resolution: {integrity: sha512-jn7/7ky30idSkd/O5yDBfAnVt+JJpepofP/POZ1iMOxK59cOfqIgg/Dj0eFsjOTMw+4ycJN0uhZH/Eb0bs/EUA==} + + '@sideway/address@4.1.5': resolution: {integrity: sha512-IqO/DUQHUkPeixNQ8n0JA6102hT9CmaljNTPmQ1u8MEhBo/R4Q8eKLN/vGZxuebwOroDB4cbpjheD4+/sKFK4Q==} - requiresBuild: true - dependencies: - '@hapi/hoek': 9.3.0 - dev: true - optional: true - /@sideway/formula@3.0.1: + '@sideway/formula@3.0.1': resolution: {integrity: sha512-/poHZJJVjx3L+zVD6g9KgHfYnb443oi7wLu/XKojDviHy6HOEOA6z1Trk5aR1dGcmPenJEgb2sK2I80LeS3MIg==} - requiresBuild: true - dev: true - optional: true - /@sideway/pinpoint@2.0.0: + '@sideway/pinpoint@2.0.0': resolution: {integrity: sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ==} - requiresBuild: true - dev: true - optional: true - /@sinclair/typebox@0.32.35: + '@sinclair/typebox@0.32.35': resolution: {integrity: sha512-Ul3YyOTU++to8cgNkttakC0dWvpERr6RYoHO2W47DLbFvrwBDJUY31B1sImH6JZSYc4Kt4PyHtoPNu+vL2r2dA==} - requiresBuild: true - dev: true - optional: true - /@sveltejs/adapter-static@3.0.6(@sveltejs/kit@2.7.4): + '@smithy/abort-controller@3.1.6': + resolution: {integrity: sha512-0XuhuHQlEqbNQZp7QxxrFTdVWdwxch4vjxYgfInF91hZFkPxf9QDrdQka0KfxFMPqLNzSw0b95uGTrLliQUavQ==} + engines: {node: '>=16.0.0'} + + '@smithy/chunked-blob-reader-native@3.0.1': + resolution: {integrity: sha512-VEYtPvh5rs/xlyqpm5NRnfYLZn+q0SRPELbvBV+C/G7IQ+ouTuo+NKKa3ShG5OaFR8NYVMXls9hPYLTvIKKDrQ==} + + '@smithy/chunked-blob-reader@4.0.0': + resolution: {integrity: sha512-jSqRnZvkT4egkq/7b6/QRCNXmmYVcHwnJldqJ3IhVpQE2atObVJ137xmGeuGFhjFUr8gCEVAOKwSY79OvpbDaQ==} + + '@smithy/config-resolver@3.0.10': + resolution: {integrity: sha512-Uh0Sz9gdUuz538nvkPiyv1DZRX9+D15EKDtnQP5rYVAzM/dnYk3P8cg73jcxyOitPgT3mE3OVj7ky7sibzHWkw==} + engines: {node: '>=16.0.0'} + + '@smithy/core@2.5.1': + resolution: {integrity: sha512-DujtuDA7BGEKExJ05W5OdxCoyekcKT3Rhg1ZGeiUWaz2BJIWXjZmsG/DIP4W48GHno7AQwRsaCb8NcBgH3QZpg==} + engines: {node: '>=16.0.0'} + + '@smithy/credential-provider-imds@3.2.5': + resolution: {integrity: sha512-4FTQGAsuwqTzVMmiRVTn0RR9GrbRfkP0wfu/tXWVHd2LgNpTY0uglQpIScXK4NaEyXbB3JmZt8gfVqO50lP8wg==} + engines: {node: '>=16.0.0'} + + '@smithy/eventstream-codec@3.1.7': + resolution: {integrity: sha512-kVSXScIiRN7q+s1x7BrQtZ1Aa9hvvP9FeCqCdBxv37GimIHgBCOnZ5Ip80HLt0DhnAKpiobFdGqTFgbaJNrazA==} + + '@smithy/eventstream-serde-browser@3.0.11': + resolution: {integrity: sha512-Pd1Wnq3CQ/v2SxRifDUihvpXzirJYbbtXfEnnLV/z0OGCTx/btVX74P86IgrZkjOydOASBGXdPpupYQI+iO/6A==} + engines: {node: '>=16.0.0'} + + '@smithy/eventstream-serde-config-resolver@3.0.8': + resolution: {integrity: sha512-zkFIG2i1BLbfoGQnf1qEeMqX0h5qAznzaZmMVNnvPZz9J5AWBPkOMckZWPedGUPcVITacwIdQXoPcdIQq5FRcg==} + engines: {node: '>=16.0.0'} + + '@smithy/eventstream-serde-node@3.0.10': + resolution: {integrity: sha512-hjpU1tIsJ9qpcoZq9zGHBJPBOeBGYt+n8vfhDwnITPhEre6APrvqq/y3XMDEGUT2cWQ4ramNqBPRbx3qn55rhw==} + engines: {node: '>=16.0.0'} + + '@smithy/eventstream-serde-universal@3.0.10': + resolution: {integrity: sha512-ewG1GHbbqsFZ4asaq40KmxCmXO+AFSM1b+DcO2C03dyJj/ZH71CiTg853FSE/3SHK9q3jiYQIFjlGSwfxQ9kww==} + engines: {node: '>=16.0.0'} + + '@smithy/fetch-http-handler@3.2.9': + resolution: {integrity: sha512-hYNVQOqhFQ6vOpenifFME546f0GfJn2OiQ3M0FDmuUu8V/Uiwy2wej7ZXxFBNqdx0R5DZAqWM1l6VRhGz8oE6A==} + + '@smithy/fetch-http-handler@4.0.0': + resolution: {integrity: sha512-MLb1f5tbBO2X6K4lMEKJvxeLooyg7guq48C2zKr4qM7F2Gpkz4dc+hdSgu77pCJ76jVqFBjZczHYAs6dp15N+g==} + + '@smithy/hash-blob-browser@3.1.7': + resolution: {integrity: sha512-4yNlxVNJifPM5ThaA5HKnHkn7JhctFUHvcaz6YXxHlYOSIrzI6VKQPTN8Gs1iN5nqq9iFcwIR9THqchUCouIfg==} + + '@smithy/hash-node@3.0.8': + resolution: {integrity: sha512-tlNQYbfpWXHimHqrvgo14DrMAgUBua/cNoz9fMYcDmYej7MAmUcjav/QKQbFc3NrcPxeJ7QClER4tWZmfwoPng==} + engines: {node: '>=16.0.0'} + + '@smithy/hash-stream-node@3.1.7': + resolution: {integrity: sha512-xMAsvJ3hLG63lsBVi1Hl6BBSfhd8/Qnp8fC06kjOpJvyyCEXdwHITa5Kvdsk6gaAXLhbZMhQMIGvgUbfnJDP6Q==} + engines: {node: '>=16.0.0'} + + '@smithy/invalid-dependency@3.0.8': + resolution: {integrity: sha512-7Qynk6NWtTQhnGTTZwks++nJhQ1O54Mzi7fz4PqZOiYXb4Z1Flpb2yRvdALoggTS8xjtohWUM+RygOtB30YL3Q==} + + '@smithy/is-array-buffer@2.2.0': + resolution: {integrity: sha512-GGP3O9QFD24uGeAXYUjwSTXARoqpZykHadOmA8G5vfJPK0/DC67qa//0qvqrJzL1xc8WQWX7/yc7fwudjPHPhA==} + engines: {node: '>=14.0.0'} + + '@smithy/is-array-buffer@3.0.0': + resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} + engines: {node: '>=16.0.0'} + + '@smithy/md5-js@3.0.8': + resolution: {integrity: sha512-LwApfTK0OJ/tCyNUXqnWCKoE2b4rDSr4BJlDAVCkiWYeHESr+y+d5zlAanuLW6fnitVJRD/7d9/kN/ZM9Su4mA==} + + '@smithy/middleware-content-length@3.0.10': + resolution: {integrity: sha512-T4dIdCs1d/+/qMpwhJ1DzOhxCZjZHbHazEPJWdB4GDi2HjIZllVzeBEcdJUN0fomV8DURsgOyrbEUzg3vzTaOg==} + engines: {node: '>=16.0.0'} + + '@smithy/middleware-endpoint@3.2.1': + resolution: {integrity: sha512-wWO3xYmFm6WRW8VsEJ5oU6h7aosFXfszlz3Dj176pTij6o21oZnzkCLzShfmRaaCHDkBXWBdO0c4sQAvLFP6zA==} + engines: {node: '>=16.0.0'} + + '@smithy/middleware-retry@3.0.25': + resolution: {integrity: sha512-m1F70cPaMBML4HiTgCw5I+jFNtjgz5z5UdGnUbG37vw6kh4UvizFYjqJGHvicfgKMkDL6mXwyPp5mhZg02g5sg==} + engines: {node: '>=16.0.0'} + + '@smithy/middleware-serde@3.0.8': + resolution: {integrity: sha512-Xg2jK9Wc/1g/MBMP/EUn2DLspN8LNt+GMe7cgF+Ty3vl+Zvu+VeZU5nmhveU+H8pxyTsjrAkci8NqY6OuvZnjA==} + engines: {node: '>=16.0.0'} + + '@smithy/middleware-stack@3.0.8': + resolution: {integrity: sha512-d7ZuwvYgp1+3682Nx0MD3D/HtkmZd49N3JUndYWQXfRZrYEnCWYc8BHcNmVsPAp9gKvlurdg/mubE6b/rPS9MA==} + engines: {node: '>=16.0.0'} + + '@smithy/node-config-provider@3.1.9': + resolution: {integrity: sha512-qRHoah49QJ71eemjuS/WhUXB+mpNtwHRWQr77J/m40ewBVVwvo52kYAmb7iuaECgGTTcYxHS4Wmewfwy++ueew==} + engines: {node: '>=16.0.0'} + + '@smithy/node-http-handler@3.2.5': + resolution: {integrity: sha512-PkOwPNeKdvX/jCpn0A8n9/TyoxjGZB8WVoJmm9YzsnAgggTj4CrjpRHlTQw7dlLZ320n1mY1y+nTRUDViKi/3w==} + engines: {node: '>=16.0.0'} + + '@smithy/property-provider@3.1.8': + resolution: {integrity: sha512-ukNUyo6rHmusG64lmkjFeXemwYuKge1BJ8CtpVKmrxQxc6rhUX0vebcptFA9MmrGsnLhwnnqeH83VTU9hwOpjA==} + engines: {node: '>=16.0.0'} + + '@smithy/protocol-http@4.1.5': + resolution: {integrity: sha512-hsjtwpIemmCkm3ZV5fd/T0bPIugW1gJXwZ/hpuVubt2hEUApIoUTrf6qIdh9MAWlw0vjMrA1ztJLAwtNaZogvg==} + engines: {node: '>=16.0.0'} + + '@smithy/querystring-builder@3.0.8': + resolution: {integrity: sha512-btYxGVqFUARbUrN6VhL9c3dnSviIwBYD9Rz1jHuN1hgh28Fpv2xjU1HeCeDJX68xctz7r4l1PBnFhGg1WBBPuA==} + engines: {node: '>=16.0.0'} + + '@smithy/querystring-parser@3.0.8': + resolution: {integrity: sha512-BtEk3FG7Ks64GAbt+JnKqwuobJNX8VmFLBsKIwWr1D60T426fGrV2L3YS5siOcUhhp6/Y6yhBw1PSPxA5p7qGg==} + engines: {node: '>=16.0.0'} + + '@smithy/service-error-classification@3.0.8': + resolution: {integrity: sha512-uEC/kCCFto83bz5ZzapcrgGqHOh/0r69sZ2ZuHlgoD5kYgXJEThCoTuw/y1Ub3cE7aaKdznb+jD9xRPIfIwD7g==} + engines: {node: '>=16.0.0'} + + '@smithy/shared-ini-file-loader@3.1.9': + resolution: {integrity: sha512-/+OsJRNtoRbtsX0UpSgWVxFZLsJHo/4sTr+kBg/J78sr7iC+tHeOvOJrS5hCpVQ6sWBbhWLp1UNiuMyZhE6pmA==} + engines: {node: '>=16.0.0'} + + '@smithy/signature-v4@4.2.1': + resolution: {integrity: sha512-NsV1jF4EvmO5wqmaSzlnTVetemBS3FZHdyc5CExbDljcyJCEEkJr8ANu2JvtNbVg/9MvKAWV44kTrGS+Pi4INg==} + engines: {node: '>=16.0.0'} + + '@smithy/smithy-client@3.4.2': + resolution: {integrity: sha512-dxw1BDxJiY9/zI3cBqfVrInij6ShjpV4fmGHesGZZUiP9OSE/EVfdwdRz0PgvkEvrZHpsj2htRaHJfftE8giBA==} + engines: {node: '>=16.0.0'} + + '@smithy/types@3.6.0': + resolution: {integrity: sha512-8VXK/KzOHefoC65yRgCn5vG1cysPJjHnOVt9d0ybFQSmJgQj152vMn4EkYhGuaOmnnZvCPav/KnYyE6/KsNZ2w==} + engines: {node: '>=16.0.0'} + + '@smithy/url-parser@3.0.8': + resolution: {integrity: sha512-4FdOhwpTW7jtSFWm7SpfLGKIBC9ZaTKG5nBF0wK24aoQKQyDIKUw3+KFWCQ9maMzrgTJIuOvOnsV2lLGW5XjTg==} + + '@smithy/util-base64@3.0.0': + resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} + engines: {node: '>=16.0.0'} + + '@smithy/util-body-length-browser@3.0.0': + resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} + + '@smithy/util-body-length-node@3.0.0': + resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} + engines: {node: '>=16.0.0'} + + '@smithy/util-buffer-from@2.2.0': + resolution: {integrity: sha512-IJdWBbTcMQ6DA0gdNhh/BwrLkDR+ADW5Kr1aZmd4k3DIF6ezMV4R2NIAmT08wQJ3yUK82thHWmC/TnK/wpMMIA==} + engines: {node: '>=14.0.0'} + + '@smithy/util-buffer-from@3.0.0': + resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} + engines: {node: '>=16.0.0'} + + '@smithy/util-config-provider@3.0.0': + resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} + engines: {node: '>=16.0.0'} + + '@smithy/util-defaults-mode-browser@3.0.25': + resolution: {integrity: sha512-fRw7zymjIDt6XxIsLwfJfYUfbGoO9CmCJk6rjJ/X5cd20+d2Is7xjU5Kt/AiDt6hX8DAf5dztmfP5O82gR9emA==} + engines: {node: '>= 10.0.0'} + + '@smithy/util-defaults-mode-node@3.0.25': + resolution: {integrity: sha512-H3BSZdBDiVZGzt8TG51Pd2FvFO0PAx/A0mJ0EH8a13KJ6iUCdYnw/Dk/MdC1kTd0eUuUGisDFaxXVXo4HHFL1g==} + engines: {node: '>= 10.0.0'} + + '@smithy/util-endpoints@2.1.4': + resolution: {integrity: sha512-kPt8j4emm7rdMWQyL0F89o92q10gvCUa6sBkBtDJ7nV2+P7wpXczzOfoDJ49CKXe5CCqb8dc1W+ZdLlrKzSAnQ==} + engines: {node: '>=16.0.0'} + + '@smithy/util-hex-encoding@3.0.0': + resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} + engines: {node: '>=16.0.0'} + + '@smithy/util-middleware@3.0.8': + resolution: {integrity: sha512-p7iYAPaQjoeM+AKABpYWeDdtwQNxasr4aXQEA/OmbOaug9V0odRVDy3Wx4ci8soljE/JXQo+abV0qZpW8NX0yA==} + engines: {node: '>=16.0.0'} + + '@smithy/util-retry@3.0.8': + resolution: {integrity: sha512-TCEhLnY581YJ+g1x0hapPz13JFqzmh/pMWL2KEFASC51qCfw3+Y47MrTmea4bUE5vsdxQ4F6/KFbUeSz22Q1ow==} + engines: {node: '>=16.0.0'} + + '@smithy/util-stream@3.2.1': + resolution: {integrity: sha512-R3ufuzJRxSJbE58K9AEnL/uSZyVdHzud9wLS8tIbXclxKzoe09CRohj2xV8wpx5tj7ZbiJaKYcutMm1eYgz/0A==} + engines: {node: '>=16.0.0'} + + '@smithy/util-uri-escape@3.0.0': + resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} + engines: {node: '>=16.0.0'} + + '@smithy/util-utf8@2.3.0': + resolution: {integrity: sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A==} + engines: {node: '>=14.0.0'} + + '@smithy/util-utf8@3.0.0': + resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} + engines: {node: '>=16.0.0'} + + '@smithy/util-waiter@3.1.7': + resolution: {integrity: sha512-d5yGlQtmN/z5eoTtIYgkvOw27US2Ous4VycnXatyoImIF9tzlcpnKqQ/V7qhvJmb2p6xZne1NopCLakdTnkBBQ==} + engines: {node: '>=16.0.0'} + + '@supabase/auth-js@2.65.1': + resolution: {integrity: sha512-IA7i2Xq2SWNCNMKxwmPlHafBQda0qtnFr8QnyyBr+KaSxoXXqEzFCnQ1dGTy6bsZjVBgXu++o3qrDypTspaAPw==} + + '@supabase/functions-js@2.4.3': + resolution: {integrity: sha512-sOLXy+mWRyu4LLv1onYydq+10mNRQ4rzqQxNhbrKLTLTcdcmS9hbWif0bGz/NavmiQfPs4ZcmQJp4WqOXlR4AQ==} + + '@supabase/node-fetch@2.6.15': + resolution: {integrity: sha512-1ibVeYUacxWYi9i0cf5efil6adJ9WRyZBLivgjs+AUpewx1F3xPi7gLgaASI2SmIQxPoCEjAsLAzKPgMJVgOUQ==} + engines: {node: 4.x || >=6.0.0} + + '@supabase/postgrest-js@1.16.3': + resolution: {integrity: sha512-HI6dsbW68AKlOPofUjDTaosiDBCtW4XAm0D18pPwxoW3zKOE2Ru13Z69Wuys9fd6iTpfDViNco5sgrtnP0666A==} + + '@supabase/realtime-js@2.10.7': + resolution: {integrity: sha512-OLI0hiSAqQSqRpGMTUwoIWo51eUivSYlaNBgxsXZE7PSoWh12wPRdVt0psUMaUzEonSB85K21wGc7W5jHnT6uA==} + + '@supabase/storage-js@2.7.1': + resolution: {integrity: sha512-asYHcyDR1fKqrMpytAS1zjyEfvxuOIp1CIXX7ji4lHHcJKqyk+sLl/Vxgm4sN6u8zvuUtae9e4kDxQP2qrwWBA==} + + '@supabase/supabase-js@2.46.1': + resolution: {integrity: sha512-HiBpd8stf7M6+tlr+/82L8b2QmCjAD8ex9YdSAKU+whB/SHXXJdus1dGlqiH9Umy9ePUuxaYmVkGd9BcvBnNvg==} + + '@sveltejs/adapter-static@3.0.6': resolution: {integrity: sha512-MGJcesnJWj7FxDcB/GbrdYD3q24Uk0PIL4QIX149ku+hlJuj//nxUbb0HxUTpjkecWfHjVveSUnUaQWnPRXlpg==} peerDependencies: '@sveltejs/kit': ^2.0.0 - dependencies: - '@sveltejs/kit': 2.7.4(@sveltejs/vite-plugin-svelte@4.0.0)(svelte@5.1.9)(vite@5.4.10) - dev: true - /@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0)(svelte@5.1.9)(vite@5.4.10): + '@sveltejs/kit@2.7.4': resolution: {integrity: sha512-3DOPQYck3CpAmPgGq/HuhJMCCz8GF0ukbompPJQ2zShoSzrEKW9iG/l0nZmaxMvuOO3NNLmZj8F3W9uzqmkNdw==} engines: {node: '>=18.13'} hasBin: true - requiresBuild: true peerDependencies: '@sveltejs/vite-plugin-svelte': ^3.0.0 || ^4.0.0-next.1 svelte: ^4.0.0 || ^5.0.0-next.0 vite: ^5.0.3 - dependencies: - '@sveltejs/vite-plugin-svelte': 4.0.0(svelte@5.1.9)(vite@5.4.10) - '@types/cookie': 0.6.0 - cookie: 0.6.0 - devalue: 5.1.1 - esm-env: 1.1.4 - import-meta-resolve: 4.1.0 - kleur: 4.1.5 - magic-string: 0.30.12 - mrmime: 2.0.0 - sade: 1.8.1 - set-cookie-parser: 2.7.1 - sirv: 3.0.0 - svelte: 5.1.9 - tiny-glob: 0.2.9 - vite: 5.4.10 - dev: true - /@sveltejs/vite-plugin-svelte-inspector@3.0.1(@sveltejs/vite-plugin-svelte@4.0.0)(svelte@5.1.9)(vite@5.4.10): + '@sveltejs/vite-plugin-svelte-inspector@3.0.1': resolution: {integrity: sha512-2CKypmj1sM4GE7HjllT7UKmo4Q6L5xFRd7VMGEWhYnZ+wc6AUVU01IBd7yUi6WnFndEwWoMNOd6e8UjoN0nbvQ==} engines: {node: ^18.0.0 || ^20.0.0 || >=22} peerDependencies: '@sveltejs/vite-plugin-svelte': ^4.0.0-next.0||^4.0.0 svelte: ^5.0.0-next.96 || ^5.0.0 vite: ^5.0.0 - dependencies: - '@sveltejs/vite-plugin-svelte': 4.0.0(svelte@5.1.9)(vite@5.4.10) - debug: 4.3.7 - svelte: 5.1.9 - vite: 5.4.10 - transitivePeerDependencies: - - supports-color - dev: true - /@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10): + '@sveltejs/vite-plugin-svelte@4.0.0': resolution: {integrity: sha512-kpVJwF+gNiMEsoHaw+FJL76IYiwBikkxYU83+BpqQLdVMff19KeRKLd2wisS8niNBMJ2omv5gG+iGDDwd8jzag==} engines: {node: ^18.0.0 || ^20.0.0 || >=22} peerDependencies: svelte: ^5.0.0-next.96 || ^5.0.0 vite: ^5.0.0 - dependencies: - '@sveltejs/vite-plugin-svelte-inspector': 3.0.1(@sveltejs/vite-plugin-svelte@4.0.0)(svelte@5.1.9)(vite@5.4.10) - debug: 4.3.7 - deepmerge: 4.3.1 - kleur: 4.1.5 - magic-string: 0.30.12 - svelte: 5.1.9 - vite: 5.4.10 - vitefu: 1.0.3(vite@5.4.10) - transitivePeerDependencies: - - supports-color - dev: true - /@swc/helpers@0.5.13: + '@swc/helpers@0.5.13': resolution: {integrity: sha512-UoKGxQ3r5kYI9dALKJapMmuK+1zWM/H17Z1+iwnNmzcJRnfFuevZs375TA5rW31pu4BS4NoSy1fRsexDXfWn5w==} - dependencies: - tslib: 2.8.1 - dev: true - /@tailwindcss/aspect-ratio@0.4.2(tailwindcss@3.4.14): + '@tailwindcss/aspect-ratio@0.4.2': resolution: {integrity: sha512-8QPrypskfBa7QIMuKHg2TA7BqES6vhBrDLOv8Unb6FcFyd3TjKbc6lcmb9UPQHxfl24sXoJ41ux/H7qQQvfaSQ==} peerDependencies: tailwindcss: '>=2.0.0 || >=3.0.0 || >=3.0.0-alpha.1' - dependencies: - tailwindcss: 3.4.14 - dev: true - /@tailwindcss/container-queries@0.1.1(tailwindcss@3.4.14): + '@tailwindcss/container-queries@0.1.1': resolution: {integrity: sha512-p18dswChx6WnTSaJCSGx6lTmrGzNNvm2FtXmiO6AuA1V4U5REyoqwmT6kgAsIMdjo07QdAfYXHJ4hnMtfHzWgA==} peerDependencies: tailwindcss: '>=3.2.0' - dependencies: - tailwindcss: 3.4.14 - dev: true - /@tailwindcss/forms@0.5.9(tailwindcss@3.4.14): + '@tailwindcss/forms@0.5.9': resolution: {integrity: sha512-tM4XVr2+UVTxXJzey9Twx48c1gcxFStqn1pQz0tRsX8o3DvxhN5oY5pvyAbUx7VTaZxpej4Zzvc6h+1RJBzpIg==} peerDependencies: tailwindcss: '>=3.0.0 || >= 3.0.0-alpha.1 || >= 4.0.0-alpha.20' - dependencies: - mini-svg-data-uri: 1.4.4 - tailwindcss: 3.4.14 - dev: true - /@tailwindcss/typography@0.5.15(tailwindcss@3.4.14): + '@tailwindcss/typography@0.5.15': resolution: {integrity: sha512-AqhlCXl+8grUz8uqExv5OTtgpjuVIwFTSXTrh8y9/pw6q2ek7fJ+Y8ZEVw7EB2DCcuCOtEjf9w3+J3rzts01uA==} peerDependencies: tailwindcss: '>=3.0.0 || insiders || >=4.0.0-alpha.20' - dependencies: - lodash.castarray: 4.4.0 - lodash.isplainobject: 4.0.6 - lodash.merge: 4.6.2 - postcss-selector-parser: 6.0.10 - tailwindcss: 3.4.14 - dev: true - /@tauri-apps/api@2.0.3: + '@tauri-apps/api@2.0.1': + resolution: {integrity: sha512-eoQWT+Tq1qSwQpHV+nw1eNYe5B/nm1PoRjQCRiEOS12I1b+X4PUcREfXVX8dPcBT6GrzWGDtaecY0+1p0Rfqlw==} + + '@tauri-apps/api@2.0.3': resolution: {integrity: sha512-840qk6n8rbXBXMA5/aAgTYsg5JAubKO0nXw5wf7IzGnUuYKGbB4oFBIZtXOIWy+E0kNTDI3qhq5iqsoMJfwp8g==} - dev: false - /@tauri-apps/cli-darwin-arm64@2.0.4: + '@tauri-apps/cli-darwin-arm64@2.0.4': resolution: {integrity: sha512-siH7rOHobb16rPbc11k64p1mxIpiRCkWmzs2qmL5IX21Gx9K5onI3Tk67Oqpf2uNupbYzItrOttaDT4NHFC7tw==} engines: {node: '>= 10'} cpu: [arm64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-darwin-x64@2.0.4: + '@tauri-apps/cli-darwin-x64@2.0.4': resolution: {integrity: sha512-zIccfbCoZMfmUpnk6PFCV0keFyfVj1A9XV3Oiiitj/dkTZ9CQvzjhX3XC0XcK4rsTWegfr2PjSrK06aiPAROAw==} engines: {node: '>= 10'} cpu: [x64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-linux-arm-gnueabihf@2.0.4: + '@tauri-apps/cli-linux-arm-gnueabihf@2.0.4': resolution: {integrity: sha512-fgQqJzefOGWCBNg4yrVA82Rg4s1XQr5K0dc2rCxBhJfa696e8dQ1LDrnWq/AiO5r+uHfVaoQTIUvxxpFicYRSA==} engines: {node: '>= 10'} cpu: [arm] os: [linux] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-linux-arm64-gnu@2.0.4: + '@tauri-apps/cli-linux-arm64-gnu@2.0.4': resolution: {integrity: sha512-u8wbt5tPA9pI6j+d7jGrfOz9UVCiTp+IYzKNiIqlrDsAjqAUFaNXYHKqOUboeFWEmI4zoCWj6LgpS2OJTQ5FKg==} engines: {node: '>= 10'} cpu: [arm64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-linux-arm64-musl@2.0.4: + '@tauri-apps/cli-linux-arm64-musl@2.0.4': resolution: {integrity: sha512-hntF1V8e3V1hlrESm93PsghDhf3lA5pbvFrRfYxU1c+fVD/jRXGVw8BH3O1lW8MWwhEg1YdhKk01oAgsuHLuig==} engines: {node: '>= 10'} cpu: [arm64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-linux-x64-gnu@2.0.4: + '@tauri-apps/cli-linux-x64-gnu@2.0.4': resolution: {integrity: sha512-Iq1GGJb+oT1T0ZV8izrgf0cBtlzPCJaWcNueRbf1ZXquMf+FSTyQv+/Lo8rq5T6buOIJOH7cAOTuEWWqiCZteg==} engines: {node: '>= 10'} cpu: [x64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-linux-x64-musl@2.0.4: + '@tauri-apps/cli-linux-x64-musl@2.0.4': resolution: {integrity: sha512-9NTk6Pf0bSwXqCBdAA+PDYts9HeHebZzIo8mbRzRyUbER6QngG5HZb9Ka36Z1QWtJjdRy6uxSb4zb/9NuTeHfA==} engines: {node: '>= 10'} cpu: [x64] os: [linux] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-win32-arm64-msvc@2.0.4: + '@tauri-apps/cli-win32-arm64-msvc@2.0.4': resolution: {integrity: sha512-OF2e9oxiBFR8A8wVMOhUx9QGN/I1ZkquWC7gVQBnA56nx9PabJlDT08QBy5UD8USqZFVznnfNr2ehlheQahb3g==} engines: {node: '>= 10'} cpu: [arm64] os: [win32] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-win32-ia32-msvc@2.0.4: + '@tauri-apps/cli-win32-ia32-msvc@2.0.4': resolution: {integrity: sha512-T+hCKB3rFP6q0saHHtR02hm6wr1ZPJ0Mkii3oRTxjPG6BBXoVzHNCYzvdgEGJPTA2sFuAQtJH764NRtNlDMifw==} engines: {node: '>= 10'} cpu: [ia32] os: [win32] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli-win32-x64-msvc@2.0.4: + '@tauri-apps/cli-win32-x64-msvc@2.0.4': resolution: {integrity: sha512-GVaiI3KWRFLomjJmApHqihhYlkJ+7FqhumhVfBO6Z2tWzZjQyVQgTdNp0kYEuW2WoAYEj0dKY6qd4YM33xYcUA==} engines: {node: '>= 10'} cpu: [x64] os: [win32] - requiresBuild: true - dev: true - optional: true - /@tauri-apps/cli@2.0.4: + '@tauri-apps/cli@2.0.4': resolution: {integrity: sha512-Hl9eFXz+O366+6su9PfaSzu2EJdFe1p8K8ghkWmi40dz8VmSE7vsMTaOStD0I71ckSOkh2ICDX7FQTBgjlpjWw==} engines: {node: '>= 10'} hasBin: true - optionalDependencies: - '@tauri-apps/cli-darwin-arm64': 2.0.4 - '@tauri-apps/cli-darwin-x64': 2.0.4 - '@tauri-apps/cli-linux-arm-gnueabihf': 2.0.4 - '@tauri-apps/cli-linux-arm64-gnu': 2.0.4 - '@tauri-apps/cli-linux-arm64-musl': 2.0.4 - '@tauri-apps/cli-linux-x64-gnu': 2.0.4 - '@tauri-apps/cli-linux-x64-musl': 2.0.4 - '@tauri-apps/cli-win32-arm64-msvc': 2.0.4 - '@tauri-apps/cli-win32-ia32-msvc': 2.0.4 - '@tauri-apps/cli-win32-x64-msvc': 2.0.4 - dev: true - /@tauri-apps/plugin-shell@2.0.1: + '@tauri-apps/plugin-deep-link@2.0.0': + resolution: {integrity: sha512-cDa2k1OrRU5DoKc0IXl1Y8RlFOU107u2phdZfT7FkApsC6TL/VAPs3YOUTT8p9/PZ50EjOKP104HFMqVqnQ0bw==} + + '@tauri-apps/plugin-dialog@2.0.1': + resolution: {integrity: sha512-fnUrNr6EfvTqdls/ufusU7h6UbNFzLKvHk/zTuOiBq01R3dTODqwctZlzakdbfSp/7pNwTKvgKTAgl/NAP/Z0Q==} + + '@tauri-apps/plugin-fs@2.0.1': + resolution: {integrity: sha512-PkeZG2WAob9Xpmr66aPvj+McDVgFjV2a7YBzYVZjiCvbGeMs6Yk09tlXhCe3EyZdT/pwWMSi8lXUace+hlsjsw==} + + '@tauri-apps/plugin-global-shortcut@2.0.0': + resolution: {integrity: sha512-pnB4CUwFVjg4twtBSxoLJ4uLFTYxsvOdC1zIbG581pYzhYatOl6mjB+ijD5SSXgiS/jNoqMcfkOF9PWAisurew==} + + '@tauri-apps/plugin-http@2.0.1': + resolution: {integrity: sha512-j6IA3pVBybSCwPpsihpX4z8bs6PluuGtr06ahL/xy4D8HunNBTmRmadJrFOQi0gOAbaig4MkQ15nzNLBLy8R1A==} + + '@tauri-apps/plugin-log@2.0.0': + resolution: {integrity: sha512-C+NII9vzswqnOQE8k7oRtnaw0z5TZsMmnirRhXkCKDEhQQH9841Us/PC1WHtGiAaJ8za1A1JB2xXndEq/47X/w==} + + '@tauri-apps/plugin-notification@2.0.0': + resolution: {integrity: sha512-6qEDYJS7mgXZWLXA0EFL+DVCJh8sJlzSoyw6B50pxhLPVFjc5Vr5DVzl5W3mUHaYhod5wsC984eQnlCCGqxYDA==} + + '@tauri-apps/plugin-os@2.0.0': + resolution: {integrity: sha512-M7hG/nNyQYTJxVG/UhTKhp9mpXriwWzrs9mqDreB8mIgqA3ek5nHLdwRZJWhkKjZrnDT4v9CpA9BhYeplTlAiA==} + + '@tauri-apps/plugin-process@2.0.0': + resolution: {integrity: sha512-OYzi0GnkrF4NAnsHZU7U3tjSoP0PbeAlO7T1Z+vJoBUH9sFQ1NSLqWYWQyf8hcb3gVWe7P1JggjiskO+LST1ug==} + + '@tauri-apps/plugin-shell@2.0.1': resolution: {integrity: sha512-akU1b77sw3qHiynrK0s930y8zKmcdrSD60htjH+mFZqv5WaakZA/XxHR3/sF1nNv9Mgmt/Shls37HwnOr00aSw==} - dependencies: - '@tauri-apps/api': 2.0.3 - dev: false - /@types/cookie@0.6.0: + '@tauri-apps/plugin-store@2.1.0': + resolution: {integrity: sha512-GADqrc17opUKYIAKnGHIUgEeTZ2wJGu1ZITKQ1WMuOFdv8fvXRFBAqsqPjE3opgWohbczX6e1NpwmZK1AnuWVw==} + + '@tauri-apps/plugin-updater@2.0.0': + resolution: {integrity: sha512-N0cl71g7RPr7zK2Fe5aoIwzw14NcdLcz7XMGFWZVjprsqgDRWoxbnUkknyCQMZthjhGkppCd/wN2MIsUz+eAhQ==} + + '@tauri-apps/plugin-upload@2.0.0': + resolution: {integrity: sha512-MdBZn4RCVmFDjuC2rbpFGjAZAbePkxrc9T35+SKJJgfXbXYw0YOzvUTsDiU7F3NzDWepO+6EbVN3Upzi9rqt/A==} + + '@ts-graphviz/adapter@2.0.5': + resolution: {integrity: sha512-K/xd2SJskbSLcUz9uYW9IDy26I3Oyutj/LREjJgcuLMxT3um4sZfy9LiUhGErHjxLRaNcaDVGSsmWeiNuhidXg==} + engines: {node: '>=18'} + + '@ts-graphviz/ast@2.0.5': + resolution: {integrity: sha512-HVT+Bn/smDzmKNJFccwgrpJaEUMPzXQ8d84JcNugzTHNUVgxAIe2Vbf4ug351YJpowivQp6/N7XCluQMjtgi5w==} + engines: {node: '>=18'} + + '@ts-graphviz/common@2.1.4': + resolution: {integrity: sha512-PNEzOgE4vgvorp/a4Ev26jVNtiX200yODoyPa8r6GfpPZbxWKW6bdXF6xWqzMkQoO1CnJOYJx2VANDbGqCqCCw==} + engines: {node: '>=18'} + + '@ts-graphviz/core@2.0.5': + resolution: {integrity: sha512-YwaCGAG3Hs0nhxl+2lVuwuTTAK3GO2XHqOGvGIwXQB16nV858rrR5w2YmWCw9nhd11uLTStxLsCAhI9koWBqDA==} + engines: {node: '>=18'} + + '@types/bun@1.1.13': + resolution: {integrity: sha512-KmQxSBgVWCl6RSuerlLGZlIWfdxkKqat0nxN61+qu4y1KDn0Ll3j7v1Pl8GnaL3a/U6GGWVTJh75ap62kR1E8Q==} + + '@types/cookie@0.6.0': resolution: {integrity: sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA==} - dev: true - /@types/estree@1.0.5: - resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==} - - /@types/estree@1.0.6: + '@types/estree@1.0.6': resolution: {integrity: sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==} - /@types/json-schema@7.0.15: + '@types/hast@3.0.4': + resolution: {integrity: sha512-WPs+bbQw5aCj+x6laNGWLH3wviHtoCv/P3+otBhbOhJgG8qtpdAMlTCxLtsTWA7LH1Oh/bFCHsBn0TPS5m30EQ==} + + '@types/json-schema@7.0.15': resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} - requiresBuild: true - dev: true - optional: true - /@types/validator@13.12.2: + '@types/lodash@4.17.13': + resolution: {integrity: sha512-lfx+dftrEZcdBPczf9d0Qv0x+j/rfNCMuC6OcfXmO8gkfeNAY88PgKUbvG56whcN23gc27yenwF6oJZXGFpYxg==} + + '@types/madge@5.0.3': + resolution: {integrity: sha512-NlQJd0qRAoyu+pawTDhLxkW940QT2dqASfwd2g/xEZu2F4Xjwa7TVRSPdbmZwUF1ygvAh0/nepeN7JjwEuOXCA==} + + '@types/mdast@4.0.4': + resolution: {integrity: sha512-kGaNbPh1k7AFzgpud/gMdvIm5xuECykRR+JnWKQno9TAXVa6WIVCGTPvYGekIDL4uwCZQSYbUxNBSb1aUo79oA==} + + '@types/node@12.20.55': + resolution: {integrity: sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==} + + '@types/node@20.12.14': + resolution: {integrity: sha512-scnD59RpYD91xngrQQLGkE+6UrHUPzeKZWhhjBSa3HSkwjbQc38+q3RoIVEwxQGRw3M+j5hpNAM+lgV3cVormg==} + + '@types/node@22.8.7': + resolution: {integrity: sha512-LidcG+2UeYIWcMuMUpBKOnryBWG/rnmOHQR5apjn8myTQcx3rinFRn7DcIFhMnS0PPFSC6OafdIKEad0lj6U0Q==} + + '@types/phoenix@1.6.5': + resolution: {integrity: sha512-xegpDuR+z0UqG9fwHqNoy3rI7JDlvaPh2TY47Fl80oq6g+hXT+c/LEuE43X48clZ6lOfANl5WrPur9fYO1RJ/w==} + + '@types/semver@7.5.8': + resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==} + + '@types/unist@3.0.3': + resolution: {integrity: sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q==} + + '@types/validator@13.12.2': resolution: {integrity: sha512-6SlHBzUW8Jhf3liqrGGXyTJSIFe4nqlJ5A5KaMZ2l/vbM3Wh3KSybots/wfWVzNLK4D1NZluDlSQIbIEPx6oyA==} - requiresBuild: true - dev: true - optional: true - /@typeschema/class-validator@0.3.0(class-validator@0.14.1): + '@types/ws@8.5.13': + resolution: {integrity: sha512-osM/gWBTPKgHV8XkTunnegTRIsvF6owmf5w+JtAfOw472dptdm0dlGv4xCt6GwQRcC2XVOvvRE/0bAoQcL2QkA==} + + '@typeschema/class-validator@0.3.0': resolution: {integrity: sha512-OJSFeZDIQ8EK1HTljKLT5CItM2wsbgczLN8tMEfz3I1Lmhc5TBfkZ0eikFzUC16tI3d1Nag7um6TfCgp2I2Bww==} - requiresBuild: true peerDependencies: class-validator: ^0.14.1 peerDependenciesMeta: class-validator: optional: true - dependencies: - '@typeschema/core': 0.14.0 - class-validator: 0.14.1 - transitivePeerDependencies: - - '@types/json-schema' - dev: true - optional: true - /@typeschema/core@0.14.0: + '@typeschema/core@0.14.0': resolution: {integrity: sha512-Ia6PtZHcL3KqsAWXjMi5xIyZ7XMH4aSnOQes8mfMLx+wGFGtGRNlwe6Y7cYvX+WfNK67OL0/HSe9t8QDygV0/w==} - requiresBuild: true peerDependencies: '@types/json-schema': ^7.0.15 peerDependenciesMeta: '@types/json-schema': optional: true - dev: true - optional: true - /@typescript-eslint/eslint-plugin@7.18.0(@typescript-eslint/parser@7.18.0)(eslint@8.57.1)(typescript@5.5.4): + '@typescript-eslint/eslint-plugin@7.18.0': resolution: {integrity: sha512-94EQTWZ40mzBc42ATNIBimBEDltSJ9RQHCC8vc/PDbxi4k8dVwUAv4o98dk50M1zB+JGFxp43FP7f8+FP8R6Sw==} engines: {node: ^18.18.0 || >=20.0.0} peerDependencies: @@ -1266,24 +2027,8 @@ packages: peerDependenciesMeta: typescript: optional: true - dependencies: - '@eslint-community/regexpp': 4.12.1 - '@typescript-eslint/parser': 7.18.0(eslint@8.57.1)(typescript@5.5.4) - '@typescript-eslint/scope-manager': 7.18.0 - '@typescript-eslint/type-utils': 7.18.0(eslint@8.57.1)(typescript@5.5.4) - '@typescript-eslint/utils': 7.18.0(eslint@8.57.1)(typescript@5.5.4) - '@typescript-eslint/visitor-keys': 7.18.0 - eslint: 8.57.1 - graphemer: 1.4.0 - ignore: 5.3.2 - natural-compare: 1.4.0 - ts-api-utils: 1.4.0(typescript@5.5.4) - typescript: 5.5.4 - transitivePeerDependencies: - - supports-color - dev: false - /@typescript-eslint/parser@7.18.0(eslint@8.57.1)(typescript@5.5.4): + '@typescript-eslint/parser@7.18.0': resolution: {integrity: sha512-4Z+L8I2OqhZV8qA132M4wNL30ypZGYOQVBfMgxDH/K5UX0PNqTu1c6za9ST5r9+tavvHiTWmBnKzpCJ/GlVFtg==} engines: {node: ^18.18.0 || >=20.0.0} peerDependencies: @@ -1292,27 +2037,12 @@ packages: peerDependenciesMeta: typescript: optional: true - dependencies: - '@typescript-eslint/scope-manager': 7.18.0 - '@typescript-eslint/types': 7.18.0 - '@typescript-eslint/typescript-estree': 7.18.0(typescript@5.5.4) - '@typescript-eslint/visitor-keys': 7.18.0 - debug: 4.3.7 - eslint: 8.57.1 - typescript: 5.5.4 - transitivePeerDependencies: - - supports-color - dev: false - /@typescript-eslint/scope-manager@7.18.0: + '@typescript-eslint/scope-manager@7.18.0': resolution: {integrity: sha512-jjhdIE/FPF2B7Z1uzc6i3oWKbGcHb87Qw7AWj6jmEqNOfDFbJWtjt/XfwCpvNkpGWlcJaog5vTR+VV8+w9JflA==} engines: {node: ^18.18.0 || >=20.0.0} - dependencies: - '@typescript-eslint/types': 7.18.0 - '@typescript-eslint/visitor-keys': 7.18.0 - dev: false - /@typescript-eslint/type-utils@7.18.0(eslint@8.57.1)(typescript@5.5.4): + '@typescript-eslint/type-utils@7.18.0': resolution: {integrity: sha512-XL0FJXuCLaDuX2sYqZUUSOJ2sG5/i1AAze+axqmLnSkNEVMVYLF+cbwlB2w8D1tinFuSikHmFta+P+HOofrLeA==} engines: {node: ^18.18.0 || >=20.0.0} peerDependencies: @@ -1321,23 +2051,12 @@ packages: peerDependenciesMeta: typescript: optional: true - dependencies: - '@typescript-eslint/typescript-estree': 7.18.0(typescript@5.5.4) - '@typescript-eslint/utils': 7.18.0(eslint@8.57.1)(typescript@5.5.4) - debug: 4.3.7 - eslint: 8.57.1 - ts-api-utils: 1.4.0(typescript@5.5.4) - typescript: 5.5.4 - transitivePeerDependencies: - - supports-color - dev: false - /@typescript-eslint/types@7.18.0: + '@typescript-eslint/types@7.18.0': resolution: {integrity: sha512-iZqi+Ds1y4EDYUtlOOC+aUmxnE9xS/yCigkjA7XpTKV6nCBd3Hp/PRGGmdwnfkV2ThMyYldP1wRpm/id99spTQ==} engines: {node: ^18.18.0 || >=20.0.0} - dev: false - /@typescript-eslint/typescript-estree@7.18.0(typescript@5.5.4): + '@typescript-eslint/typescript-estree@7.18.0': resolution: {integrity: sha512-aP1v/BSPnnyhMHts8cf1qQ6Q1IFwwRvAQGRvBFkWlo3/lH29OXA3Pts+c10nxRxIBrDnoMqzhgdwVe5f2D6OzA==} engines: {node: ^18.18.0 || >=20.0.0} peerDependencies: @@ -1345,360 +2064,329 @@ packages: peerDependenciesMeta: typescript: optional: true - dependencies: - '@typescript-eslint/types': 7.18.0 - '@typescript-eslint/visitor-keys': 7.18.0 - debug: 4.3.7 - globby: 11.1.0 - is-glob: 4.0.3 - minimatch: 9.0.5 - semver: 7.6.3 - ts-api-utils: 1.4.0(typescript@5.5.4) - typescript: 5.5.4 - transitivePeerDependencies: - - supports-color - dev: false - /@typescript-eslint/utils@7.18.0(eslint@8.57.1)(typescript@5.5.4): + '@typescript-eslint/utils@7.18.0': resolution: {integrity: sha512-kK0/rNa2j74XuHVcoCZxdFBMF+aq/vH83CXAOHieC+2Gis4mF8jJXT5eAfyD3K0sAxtPuwxaIOIOvhwzVDt/kw==} engines: {node: ^18.18.0 || >=20.0.0} peerDependencies: eslint: ^8.56.0 - dependencies: - '@eslint-community/eslint-utils': 4.4.1(eslint@8.57.1) - '@typescript-eslint/scope-manager': 7.18.0 - '@typescript-eslint/types': 7.18.0 - '@typescript-eslint/typescript-estree': 7.18.0(typescript@5.5.4) - eslint: 8.57.1 - transitivePeerDependencies: - - supports-color - - typescript - dev: false - /@typescript-eslint/visitor-keys@7.18.0: + '@typescript-eslint/visitor-keys@7.18.0': resolution: {integrity: sha512-cDF0/Gf81QpY3xYyJKDV14Zwdmid5+uuENhjH2EqFaF0ni+yAyq/LzMaIJdhNJXZI7uLzwIlA+V7oWoyn6Curg==} engines: {node: ^18.18.0 || >=20.0.0} - dependencies: - '@typescript-eslint/types': 7.18.0 - eslint-visitor-keys: 3.4.3 - dev: false - /@ungap/structured-clone@1.2.0: + '@ungap/structured-clone@1.2.0': resolution: {integrity: sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==} - dev: false - /@vinejs/compiler@2.5.0: + '@vinejs/compiler@2.5.0': resolution: {integrity: sha512-hg4ekaB5Y2zh+IWzBiC/WCDWrIfpVnKu/ubUvelKlidc/VbulsexoFRw5kJGHZenPVI5YzNnDeTdYSALkTV7jQ==} engines: {node: '>=18.0.0'} - requiresBuild: true - dev: true - optional: true - /@vinejs/vine@1.8.0: + '@vinejs/vine@1.8.0': resolution: {integrity: sha512-Qq3XxbA26jzqS9ICifkqzT399lMQZ2fWtqeV3luI2as+UIK7qDifJFU2Q4W3q3IB5VXoWxgwAZSZEO0em9I/qQ==} engines: {node: '>=18.16.0'} - requiresBuild: true - dependencies: - '@poppinss/macroable': 1.0.3 - '@types/validator': 13.12.2 - '@vinejs/compiler': 2.5.0 - camelcase: 8.0.0 - dayjs: 1.11.13 - dlv: 1.1.3 - normalize-url: 8.0.1 - validator: 13.12.0 - dev: true - optional: true - /acorn-jsx@5.3.2(acorn@8.12.1): + '@vue/compiler-core@3.5.12': + resolution: {integrity: sha512-ISyBTRMmMYagUxhcpyEH0hpXRd/KqDU4ymofPgl2XAkY9ZhQ+h0ovEZJIiPop13UmR/54oA2cgMDjgroRelaEw==} + + '@vue/compiler-dom@3.5.12': + resolution: {integrity: sha512-9G6PbJ03uwxLHKQ3P42cMTi85lDRvGLB2rSGOiQqtXELat6uI4n8cNz9yjfVHRPIu+MsK6TE418Giruvgptckg==} + + '@vue/compiler-sfc@3.5.12': + resolution: {integrity: sha512-2k973OGo2JuAa5+ZlekuQJtitI5CgLMOwgl94BzMCsKZCX/xiqzJYzapl4opFogKHqwJk34vfsaKpfEhd1k5nw==} + + '@vue/compiler-ssr@3.5.12': + resolution: {integrity: sha512-eLwc7v6bfGBSM7wZOGPmRavSWzNFF6+PdRhE+VFJhNCgHiF8AM7ccoqcv5kBXA2eWUfigD7byekvf/JsOfKvPA==} + + '@vue/shared@3.5.12': + resolution: {integrity: sha512-L2RPSAwUFbgZH20etwrXyVyCBu9OxRSi8T/38QsvnkJyvq2LufW2lDCOzm7t/U9C1mkhJGWYfCuFBCmIuNivrg==} + + acorn-jsx@5.3.2: resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==} peerDependencies: acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 - dependencies: - acorn: 8.12.1 - dev: false - /acorn-typescript@1.4.13(acorn@8.12.1): + acorn-typescript@1.4.13: resolution: {integrity: sha512-xsc9Xv0xlVfwp2o7sQ+GCQ1PgbkdcpWdTzrwXxO3xDMTAywVS3oXVOcOHuRjAPkS4P9b+yc/qNF15460v+jp4Q==} peerDependencies: acorn: '>=8.9.0' - dependencies: - acorn: 8.12.1 - /acorn@8.12.1: + acorn@8.12.1: resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} engines: {node: '>=0.4.0'} hasBin: true - /ajv@6.12.6: - resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} - dependencies: - fast-deep-equal: 3.1.3 - fast-json-stable-stringify: 2.1.0 - json-schema-traverse: 0.4.1 - uri-js: 4.4.1 - dev: false + agent-base@7.1.1: + resolution: {integrity: sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==} + engines: {node: '>= 14'} - /ansi-regex@5.0.1: + ajv@6.12.6: + resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} + + ansi-colors@4.1.3: + resolution: {integrity: sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==} + engines: {node: '>=6'} + + ansi-regex@5.0.1: resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} engines: {node: '>=8'} - /ansi-regex@6.1.0: + ansi-regex@6.1.0: resolution: {integrity: sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==} engines: {node: '>=12'} - dev: true - /ansi-styles@4.3.0: + ansi-styles@4.3.0: resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} engines: {node: '>=8'} - dependencies: - color-convert: 2.0.1 - /ansi-styles@6.2.1: + ansi-styles@6.2.1: resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} engines: {node: '>=12'} - dev: true - /any-promise@1.3.0: + any-promise@1.3.0: resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} - dev: true - /anymatch@3.1.3: + anymatch@3.1.3: resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} engines: {node: '>= 8'} - dependencies: - normalize-path: 3.0.0 - picomatch: 2.3.1 - dev: true - /arg@5.0.2: + app-module-path@2.2.0: + resolution: {integrity: sha512-gkco+qxENJV+8vFcDiiFhuoSvRXb2a/QPqpSoWhVz829VNJfOTnELbBmPmNKFxf3xdNnw4DWCkzkDaavcX/1YQ==} + + arg@5.0.2: resolution: {integrity: sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==} - dev: true - /argparse@2.0.1: + argparse@1.0.10: + resolution: {integrity: sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==} + + argparse@2.0.1: resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} - dev: false - /aria-query@5.3.2: + aria-query@5.3.2: resolution: {integrity: sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==} engines: {node: '>= 0.4'} - /arktype@2.0.0-rc.8: + arktype@2.0.0-rc.8: resolution: {integrity: sha512-ByrqjptsavUCUL9ptts6BUL2LCNkVZyniOdaBw76dlBQ6gYIhYSeycuuj4gRFwcAafszOnAPD2fAqHK7bbo/Zw==} - requiresBuild: true - dependencies: - '@ark/schema': 0.10.0 - '@ark/util': 0.10.0 - dev: true - optional: true - /array-union@2.1.0: + array-union@2.1.0: resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} engines: {node: '>=8'} - dev: false - /autoprefixer@10.4.20(postcss@8.4.47): + ast-module-types@6.0.0: + resolution: {integrity: sha512-LFRg7178Fw5R4FAEwZxVqiRI8IxSM+Ay2UBrHoCerXNme+kMMMfz7T3xDGV/c2fer87hcrtgJGsnSOfUrPK6ng==} + engines: {node: '>=18'} + + autoprefixer@10.4.20: resolution: {integrity: sha512-XY25y5xSv/wEoqzDyXXME4AFfkZI0P23z6Fs3YgymDnKJkCGOnkL0iTxCa85UTqaSgfcqyf3UA6+c7wUvx/16g==} engines: {node: ^10 || ^12 || >=14} hasBin: true peerDependencies: postcss: ^8.1.0 - dependencies: - browserslist: 4.24.2 - caniuse-lite: 1.0.30001676 - fraction.js: 4.3.7 - normalize-range: 0.1.2 - picocolors: 1.1.0 - postcss: 8.4.47 - postcss-value-parser: 4.2.0 - dev: true - /axobject-query@4.1.0: + axobject-query@4.1.0: resolution: {integrity: sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==} engines: {node: '>= 0.4'} - /balanced-match@1.0.2: + balanced-match@1.0.2: resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} - /binary-extensions@2.3.0: + base64-js@1.5.1: + resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} + + better-path-resolve@1.0.0: + resolution: {integrity: sha512-pbnl5XzGBdrFU/wT4jqmJVPn2B6UHPBOhzMQkY/SPUPB6QtUXtmBHBIwCbXJol93mOpGMnQyP/+BB19q04xj7g==} + engines: {node: '>=4'} + + bin-links@5.0.0: + resolution: {integrity: sha512-sdleLVfCjBtgO5cNjA2HVRvWBJAHs4zwenaCPMNJAJU0yNxpzj80IpjOIimkpkr+mhlA+how5poQtt53PygbHA==} + engines: {node: ^18.17.0 || >=20.5.0} + + binary-extensions@2.3.0: resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} engines: {node: '>=8'} - dev: true - /bits-ui@0.21.16(svelte@5.1.9): + bits-ui@0.21.16: resolution: {integrity: sha512-XFZ7/bK7j/K+5iktxX/ZpmoFHjYjpPzP5EOO/4bWiaFg5TG1iMcfjDhlBTQnJxD6BoVoHuqeZPHZvaTgF4Iv3Q==} peerDependencies: svelte: ^4.0.0 || ^5.0.0-next.118 - dependencies: - '@internationalized/date': 3.5.6 - '@melt-ui/svelte': 0.76.2(svelte@5.1.9) - nanoid: 5.0.8 - svelte: 5.1.9 - dev: true - /bits-ui@1.0.0-next.36(svelte@5.1.9): + bits-ui@1.0.0-next.36: resolution: {integrity: sha512-xkMLjRYWsS7ZwqWssE1tYqQb/7vo89tcJdznc5nEwl+wpMZNeXuUDm3hc++hFZImursN5pltD1KkVTmAfmbsaQ==} engines: {node: '>=18', pnpm: '>=8.7.0'} peerDependencies: svelte: ^5.0.0-next.1 - dependencies: - '@floating-ui/core': 1.6.8 - '@floating-ui/dom': 1.6.12 - '@internationalized/date': 3.5.6 - esm-env: 1.1.4 - runed: 0.15.3(svelte@5.1.9) - svelte: 5.1.9 - svelte-toolbelt: 0.4.6(svelte@5.1.9) - dev: true - /brace-expansion@1.1.11: + bl@4.1.0: + resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==} + + bowser@2.11.0: + resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} + + brace-expansion@1.1.11: resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} - dependencies: - balanced-match: 1.0.2 - concat-map: 0.0.1 - dev: false - /brace-expansion@2.0.1: + brace-expansion@2.0.1: resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} - dependencies: - balanced-match: 1.0.2 - /braces@3.0.3: + braces@3.0.3: resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} engines: {node: '>=8'} - dependencies: - fill-range: 7.1.1 - /browserslist@4.24.2: + browserslist@4.24.2: resolution: {integrity: sha512-ZIc+Q62revdMcqC6aChtW4jz3My3klmCO1fEmINZY/8J3EpBg5/A/D0AKmBveUh6pgoeycoMkVMko84tuYS+Gg==} engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7} hasBin: true - dependencies: - caniuse-lite: 1.0.30001676 - electron-to-chromium: 1.5.50 - node-releases: 2.0.18 - update-browserslist-db: 1.1.1(browserslist@4.24.2) - dev: true - /buffer-from@1.1.2: + buffer-from@1.1.2: resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==} - requiresBuild: true - dev: true - optional: true - /callsites@3.1.0: + buffer@5.7.1: + resolution: {integrity: sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==} + + bun-types@1.1.34: + resolution: {integrity: sha512-br5QygTEL/TwB4uQOb96Ky22j4Gq2WxWH/8Oqv20fk5HagwKXo/akB+LiYgSfzexCt6kkcUaVm+bKiPl71xPvw==} + + callsites@3.1.0: resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} engines: {node: '>=6'} - dev: false - /camelcase-css@2.0.1: + camelcase-css@2.0.1: resolution: {integrity: sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==} engines: {node: '>= 6'} - dev: true - /camelcase@8.0.0: + camelcase@8.0.0: resolution: {integrity: sha512-8WB3Jcas3swSvjIeA2yvCJ+Miyz5l1ZmB6HFb9R1317dt9LCQoswg/BGrmAmkWVEszSrrg4RwmO46qIm2OEnSA==} engines: {node: '>=16'} - requiresBuild: true - dev: true - optional: true - /caniuse-lite@1.0.30001676: + caniuse-lite@1.0.30001676: resolution: {integrity: sha512-Qz6zwGCiPghQXGJvgQAem79esjitvJ+CxSbSQkW9H/UX5hg8XM88d4lp2W+MEQ81j+Hip58Il+jGVdazk1z9cw==} - dev: true - /chalk@4.1.2: + ccount@2.0.1: + resolution: {integrity: sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg==} + + chalk@4.1.2: resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} engines: {node: '>=10'} - dependencies: - ansi-styles: 4.3.0 - supports-color: 7.2.0 - dev: false - /chokidar@3.6.0: + character-entities-html4@2.1.0: + resolution: {integrity: sha512-1v7fgQRj6hnSwFpq1Eu0ynr/CDEw0rXo2B61qXrLNdHZmPKgb7fqS1a2JwF0rISo9q77jDI8VMEHoApn8qDoZA==} + + character-entities-legacy@3.0.0: + resolution: {integrity: sha512-RpPp0asT/6ufRm//AJVwpViZbGM/MkjQFxJccQRHmISF/22NBtsHqAWmL+/pmkPWoIUJdWyeVleTl1wydHATVQ==} + + chardet@0.7.0: + resolution: {integrity: sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==} + + chokidar@3.6.0: resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} engines: {node: '>= 8.10.0'} - dependencies: - anymatch: 3.1.3 - braces: 3.0.3 - glob-parent: 5.1.2 - is-binary-path: 2.1.0 - is-glob: 4.0.3 - normalize-path: 3.0.0 - readdirp: 3.6.0 - optionalDependencies: - fsevents: 2.3.3 - dev: true - /chokidar@4.0.1: + chokidar@4.0.1: resolution: {integrity: sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==} engines: {node: '>= 14.16.0'} - dependencies: - readdirp: 4.0.2 - dev: true - /class-validator@0.14.1: + chownr@3.0.0: + resolution: {integrity: sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g==} + engines: {node: '>=18'} + + ci-info@3.9.0: + resolution: {integrity: sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==} + engines: {node: '>=8'} + + class-validator@0.14.1: resolution: {integrity: sha512-2VEG9JICxIqTpoK1eMzZqaV+u/EiwEJkMGzTrZf6sU/fwsnOITVgYJ8yojSy6CaXtO9V0Cc6ZQZ8h8m4UBuLwQ==} - requiresBuild: true - dependencies: - '@types/validator': 13.12.2 - libphonenumber-js: 1.11.12 - validator: 13.12.0 - dev: true - optional: true - /clsx@2.1.1: + cli-cursor@3.1.0: + resolution: {integrity: sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==} + engines: {node: '>=8'} + + cli-spinners@2.9.2: + resolution: {integrity: sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==} + engines: {node: '>=6'} + + clone@1.0.4: + resolution: {integrity: sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==} + engines: {node: '>=0.8'} + + clsx@2.1.1: resolution: {integrity: sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==} engines: {node: '>=6'} - dev: true - /color-convert@2.0.1: + cmd-shim@7.0.0: + resolution: {integrity: sha512-rtpaCbr164TPPh+zFdkWpCyZuKkjpAzODfaZCf/SVJZzJN+4bHQb/LP3Jzq5/+84um3XXY8r548XiWKSborwVw==} + engines: {node: ^18.17.0 || >=20.5.0} + + color-convert@2.0.1: resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} engines: {node: '>=7.0.0'} - dependencies: - color-name: 1.1.4 - /color-name@1.1.4: + color-name@1.1.4: resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} - /commander@4.1.1: + comlink-stdio@0.1.7: + resolution: {integrity: sha512-LTTyZfEq3KuDKnNSFDsDbJzNh5F6h+MTd13D6K7yRt9zvnP2nxZ5DgyaAaw2H87vQUzJ1T93Se1o3FHk9Ch7cw==} + peerDependencies: + typescript: ^5.0.0 + + comlink@4.4.1: + resolution: {integrity: sha512-+1dlx0aY5Jo1vHy/tSsIGpSkN4tS9rZSW8FIhG0JH/crs9wwweswIo/POr451r7bZww3hFbPAKnTpimzL/mm4Q==} + + comma-separated-tokens@2.0.3: + resolution: {integrity: sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg==} + + commander@12.1.0: + resolution: {integrity: sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==} + engines: {node: '>=18'} + + commander@2.20.3: + resolution: {integrity: sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==} + + commander@4.1.1: resolution: {integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==} engines: {node: '>= 6'} - dev: true - /concat-map@0.0.1: + commander@7.2.0: + resolution: {integrity: sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==} + engines: {node: '>= 10'} + + commander@9.5.0: + resolution: {integrity: sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==} + engines: {node: ^12.20.0 || >=14} + + commondir@1.0.1: + resolution: {integrity: sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==} + + concat-map@0.0.1: resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - dev: false - /convert-source-map@2.0.0: + convert-source-map@2.0.0: resolution: {integrity: sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==} - dev: true - /cookie@0.6.0: + cookie@0.6.0: resolution: {integrity: sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==} engines: {node: '>= 0.6'} - dev: true - /cross-spawn@7.0.3: + cross-spawn@5.1.0: + resolution: {integrity: sha512-pTgQJ5KC0d2hcY8eyL1IzlBPYjTkyH72XRZPnLyKus2mBfNjQs3klqbJU2VILqZryAZUt9JOb3h/mWMy23/f5A==} + + cross-spawn@7.0.3: resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} engines: {node: '>= 8'} - dependencies: - path-key: 3.1.1 - shebang-command: 2.0.0 - which: 2.0.2 - /cssesc@3.0.0: + cssesc@3.0.0: resolution: {integrity: sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==} engines: {node: '>=4'} hasBin: true - /dayjs@1.11.13: - resolution: {integrity: sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==} - requiresBuild: true - dev: true - optional: true + data-uri-to-buffer@4.0.1: + resolution: {integrity: sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==} + engines: {node: '>= 12'} - /debug@4.3.7: + dayjs@1.11.13: + resolution: {integrity: sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==} + + debug@4.3.7: resolution: {integrity: sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==} engines: {node: '>=6.0'} peerDependencies: @@ -1706,183 +2394,188 @@ packages: peerDependenciesMeta: supports-color: optional: true - dependencies: - ms: 2.1.3 - /deep-is@0.1.4: + deep-extend@0.6.0: + resolution: {integrity: sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==} + engines: {node: '>=4.0.0'} + + deep-is@0.1.4: resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} - dev: false - /deepmerge@4.3.1: + deepmerge@4.3.1: resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==} engines: {node: '>=0.10.0'} - dev: true - /dequal@2.0.3: + defaults@1.0.4: + resolution: {integrity: sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==} + + dependency-tree@11.0.1: + resolution: {integrity: sha512-eCt7HSKIC9NxgIykG2DRq3Aewn9UhVS14MB3rEn6l/AsEI1FBg6ZGSlCU0SZ6Tjm2kkhj6/8c2pViinuyKELhg==} + engines: {node: '>=18'} + hasBin: true + + dequal@2.0.3: resolution: {integrity: sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==} engines: {node: '>=6'} - dev: true - /devalue@5.1.1: + detect-indent@6.1.0: + resolution: {integrity: sha512-reYkTUJAZb9gUuZ2RvVCNhVHdg62RHnJ7WJl8ftMi4diZ6NWlciOzQN88pUhSELEwflJht4oQDv0F0BMlwaYtA==} + engines: {node: '>=8'} + + detective-amd@6.0.0: + resolution: {integrity: sha512-NTqfYfwNsW7AQltKSEaWR66hGkTeD52Kz3eRQ+nfkA9ZFZt3iifRCWh+yZ/m6t3H42JFwVFTrml/D64R2PAIOA==} + engines: {node: '>=18'} + hasBin: true + + detective-cjs@6.0.0: + resolution: {integrity: sha512-R55jTS6Kkmy6ukdrbzY4x+I7KkXiuDPpFzUViFV/tm2PBGtTCjkh9ZmTuJc1SaziMHJOe636dtiZLEuzBL9drg==} + engines: {node: '>=18'} + + detective-es6@5.0.0: + resolution: {integrity: sha512-NGTnzjvgeMW1khUSEXCzPDoraLenWbUjCFjwxReH+Ir+P6LGjYtaBbAvITWn2H0VSC+eM7/9LFOTAkrta6hNYg==} + engines: {node: '>=18'} + + detective-postcss@7.0.0: + resolution: {integrity: sha512-pSXA6dyqmBPBuERpoOKKTUUjQCZwZPLRbd1VdsTbt6W+m/+6ROl4BbE87yQBUtLoK7yX8pvXHdKyM/xNIW9F7A==} + engines: {node: ^14.0.0 || >=16.0.0} + peerDependencies: + postcss: ^8.4.38 + + detective-sass@6.0.0: + resolution: {integrity: sha512-h5GCfFMkPm4ZUUfGHVPKNHKT8jV7cSmgK+s4dgQH4/dIUNh9/huR1fjEQrblOQNDalSU7k7g+tiW9LJ+nVEUhg==} + engines: {node: '>=18'} + + detective-scss@5.0.0: + resolution: {integrity: sha512-Y64HyMqntdsCh1qAH7ci95dk0nnpA29g319w/5d/oYcHolcGUVJbIhOirOFjfN1KnMAXAFm5FIkZ4l2EKFGgxg==} + engines: {node: '>=18'} + + detective-stylus@5.0.0: + resolution: {integrity: sha512-KMHOsPY6aq3196WteVhkY5FF+6Nnc/r7q741E+Gq+Ax9mhE2iwj8Hlw8pl+749hPDRDBHZ2WlgOjP+twIG61vQ==} + engines: {node: '>=18'} + + detective-typescript@13.0.0: + resolution: {integrity: sha512-tcMYfiFWoUejSbvSblw90NDt76/4mNftYCX0SMnVRYzSXv8Fvo06hi4JOPdNvVNxRtCAKg3MJ3cBJh+ygEMH+A==} + engines: {node: ^14.14.0 || >=16.0.0} + peerDependencies: + typescript: ^5.4.4 + + detective-vue2@2.1.0: + resolution: {integrity: sha512-IHQVhwk7dKaJ+GHBsL27mS9NRO1/vLZJPSODqtJgKquij0/UL8NvrbXbADbYeTkwyh1ReW/v9u9IRyEO5dvGZg==} + engines: {node: '>=18'} + peerDependencies: + typescript: ^5.4.4 + + devalue@5.1.1: resolution: {integrity: sha512-maua5KUiapvEwiEAe+XnlZ3Rh0GD+qI1J/nb9vrJc3muPXvcF/8gXYTWF76+5DAqHyDUtOIImEuo0YKE9mshVw==} - dev: true - /didyoumean@1.2.2: + devlop@1.1.0: + resolution: {integrity: sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA==} + + didyoumean@1.2.2: resolution: {integrity: sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==} - dev: true - /dir-glob@3.0.1: + dir-glob@3.0.1: resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} engines: {node: '>=8'} - dependencies: - path-type: 4.0.0 - dev: false - /dlv@1.1.3: + dlv@1.1.3: resolution: {integrity: sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==} - dev: true - /doctrine@3.0.0: + doctrine@3.0.0: resolution: {integrity: sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==} engines: {node: '>=6.0.0'} - dependencies: - esutils: 2.0.3 - dev: false - /dotenv@16.0.3: + dotenv@16.0.3: resolution: {integrity: sha512-7GO6HghkA5fYG9TYnNxi14/7K9f5occMlp3zXAuSxn7CKCxt9xbNWG7yF8hTCSUchlfWSe3uLmlPfigevRItzQ==} engines: {node: '>=12'} - dev: false - /eastasianwidth@0.2.0: + eastasianwidth@0.2.0: resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} - dev: true - /effect@3.10.8: + effect@3.10.8: resolution: {integrity: sha512-vCKKp7EL6j7x1xfBbmndcQvooNE5FEDxkVro2yqEGBR2Ogn4I4XVK5A4eLVlDxS9egvX7EFaBwDRfm7k6IM6oA==} - requiresBuild: true - dependencies: - fast-check: 3.23.0 - dev: true - optional: true - /electron-to-chromium@1.5.50: + electron-to-chromium@1.5.50: resolution: {integrity: sha512-eMVObiUQ2LdgeO1F/ySTXsvqvxb6ZH2zPGaMYsWzRDdOddUa77tdmI0ltg+L16UpbWdhPmuF3wIQYyQq65WfZw==} - dev: true - /embla-carousel-reactive-utils@8.3.1(embla-carousel@8.3.1): + embla-carousel-reactive-utils@8.3.1: resolution: {integrity: sha512-Js6rTTINNGnUGPu7l5kTcheoSbEnP5Ak2iX0G9uOoI8okTNLMzuWlEIpYFd1WP0Sq82FFcLkKM2oiO6jcElZ/Q==} peerDependencies: embla-carousel: 8.3.1 - dependencies: - embla-carousel: 8.3.1 - dev: true - /embla-carousel-svelte@8.3.1(svelte@5.1.9): + embla-carousel-svelte@8.3.1: resolution: {integrity: sha512-dD+NpbJJEkyGL2NtXKCIkBN51Lj4V6pRh85lpXfjh690UTvJsO5Zdp6xZ6nBTOjSsQ5IczdEXvv3DrvXfyg1SA==} peerDependencies: svelte: ^3.49.0 || ^4.0.0 || ^5.0.0 - dependencies: - embla-carousel: 8.3.1 - embla-carousel-reactive-utils: 8.3.1(embla-carousel@8.3.1) - svelte: 5.1.9 - dev: true - /embla-carousel@8.3.1: + embla-carousel@8.3.1: resolution: {integrity: sha512-DutFjtEO586XptDn4cwvBJwsR/8fMa4jUk5Jk2g+/elKgu8mdn0Z2sx33g4JskvbLc1/6P8Xg4QlfELGJFcP5A==} - dev: true - /emoji-regex@8.0.0: + emoji-regex@8.0.0: resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - dev: true - /emoji-regex@9.2.2: + emoji-regex@9.2.2: resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} - dev: true - /esbuild-runner@2.2.2(esbuild@0.21.5): + enhanced-resolve@5.17.1: + resolution: {integrity: sha512-LMHl3dXhTcfv8gM4kEzIUeTQ+7fpdA0l2tUf34BddXPkz2A5xJ5L/Pchd5BL6rdccM9QGvu0sWZzK1Z1t4wwyg==} + engines: {node: '>=10.13.0'} + + enquirer@2.4.1: + resolution: {integrity: sha512-rRqJg/6gd538VHvR3PSrdRBb/1Vy2YfzHqzvbhGIQpDRKIa4FgV/54b5Q1xYSxOOwKvjXweS26E0Q+nAMwp2pQ==} + engines: {node: '>=8.6'} + + entities@4.5.0: + resolution: {integrity: sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==} + engines: {node: '>=0.12'} + + esbuild-runner@2.2.2: resolution: {integrity: sha512-fRFVXcmYVmSmtYm2mL8RlUASt2TDkGh3uRcvHFOKNr/T58VrfVeKD9uT9nlgxk96u0LS0ehS/GY7Da/bXWKkhw==} hasBin: true - requiresBuild: true peerDependencies: esbuild: '*' - dependencies: - esbuild: 0.21.5 - source-map-support: 0.5.21 - tslib: 2.4.0 - dev: true - optional: true - /esbuild@0.21.5: + esbuild@0.21.5: resolution: {integrity: sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==} engines: {node: '>=12'} hasBin: true - requiresBuild: true - 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 - dev: true - /escalade@3.2.0: + esbuild@0.24.0: + resolution: {integrity: sha512-FuLPevChGDshgSicjisSooU0cemp/sGXR841D5LHMB7mTVOmsEHcAxaH3irL53+8YDIeVNQEySh4DaYU/iuPqQ==} + engines: {node: '>=18'} + hasBin: true + + escalade@3.2.0: resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} engines: {node: '>=6'} - dev: true - /escape-string-regexp@4.0.0: + escape-string-regexp@4.0.0: resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} engines: {node: '>=10'} - dev: false - /eslint-compat-utils@0.5.1(eslint@8.57.1): + escodegen@2.1.0: + resolution: {integrity: sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w==} + engines: {node: '>=6.0'} + hasBin: true + + eslint-compat-utils@0.5.1: resolution: {integrity: sha512-3z3vFexKIEnjHE3zCMRo6fn/e44U7T1khUjg+Hp0ZQMCigh28rALD0nPFBcGZuiLC5rLZa2ubQHDRln09JfU2Q==} engines: {node: '>=12'} peerDependencies: eslint: '>=6.0.0' - dependencies: - eslint: 8.57.1 - semver: 7.6.3 - dev: false - /eslint-config-prettier@9.1.0(eslint@8.57.1): + eslint-config-prettier@9.1.0: resolution: {integrity: sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw==} hasBin: true peerDependencies: eslint: '>=7.0.0' - dependencies: - eslint: 8.57.1 - dev: false - /eslint-config-turbo@2.2.3(eslint@8.57.1): + eslint-config-turbo@2.2.3: resolution: {integrity: sha512-/zwNU+G2w0HszXzWILdl6/Catt86ejUG7vsFSdpnFzFAAUbbT2TxgoCFvC1fKtm6+SkQsXwkRRe9tFz0aMftpg==} peerDependencies: eslint: '>6.6.0' - dependencies: - eslint: 8.57.1 - eslint-plugin-turbo: 2.2.3(eslint@8.57.1) - dev: false - /eslint-plugin-svelte@2.46.0(eslint@8.57.1)(svelte@5.1.9): + eslint-plugin-svelte@2.46.0: resolution: {integrity: sha512-1A7iEMkzmCZ9/Iz+EAfOGYL8IoIG6zeKEq1SmpxGeM5SXmoQq+ZNnCpXFVJpsxPWYx8jIVGMerQMzX20cqUl0g==} engines: {node: ^14.17.0 || >=16.0.0} peerDependencies: @@ -1891,836 +2584,800 @@ packages: peerDependenciesMeta: svelte: optional: true - dependencies: - '@eslint-community/eslint-utils': 4.4.1(eslint@8.57.1) - '@jridgewell/sourcemap-codec': 1.5.0 - eslint: 8.57.1 - eslint-compat-utils: 0.5.1(eslint@8.57.1) - esutils: 2.0.3 - known-css-properties: 0.35.0 - postcss: 8.4.47 - postcss-load-config: 3.1.4(postcss@8.4.47) - postcss-safe-parser: 6.0.0(postcss@8.4.47) - postcss-selector-parser: 6.1.2 - semver: 7.6.3 - svelte: 5.1.9 - svelte-eslint-parser: 0.43.0(svelte@5.1.9) - transitivePeerDependencies: - - ts-node - dev: false - /eslint-plugin-turbo@2.2.3(eslint@8.57.1): + eslint-plugin-turbo@2.2.3: resolution: {integrity: sha512-LHt35VwxthdGVO6hQRfvmFb6ee8/exAzAYWCy4o87Bnp7urltP8qg7xMd4dPSLAhtfnI2xSo1WgeVaR3MeItxw==} peerDependencies: eslint: '>6.6.0' - dependencies: - dotenv: 16.0.3 - eslint: 8.57.1 - dev: false - /eslint-scope@7.2.2: + eslint-scope@7.2.2: resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - dependencies: - esrecurse: 4.3.0 - estraverse: 5.3.0 - dev: false - /eslint-visitor-keys@3.4.3: + eslint-visitor-keys@3.4.3: resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - dev: false - /eslint@8.57.1: + eslint@8.57.1: resolution: {integrity: sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} deprecated: This version is no longer supported. Please see https://eslint.org/version-support for other options. hasBin: true - dependencies: - '@eslint-community/eslint-utils': 4.4.1(eslint@8.57.1) - '@eslint-community/regexpp': 4.12.1 - '@eslint/eslintrc': 2.1.4 - '@eslint/js': 8.57.1 - '@humanwhocodes/config-array': 0.13.0 - '@humanwhocodes/module-importer': 1.0.1 - '@nodelib/fs.walk': 1.2.8 - '@ungap/structured-clone': 1.2.0 - ajv: 6.12.6 - chalk: 4.1.2 - cross-spawn: 7.0.3 - debug: 4.3.7 - doctrine: 3.0.0 - escape-string-regexp: 4.0.0 - eslint-scope: 7.2.2 - eslint-visitor-keys: 3.4.3 - espree: 9.6.1 - esquery: 1.6.0 - esutils: 2.0.3 - fast-deep-equal: 3.1.3 - file-entry-cache: 6.0.1 - find-up: 5.0.0 - glob-parent: 6.0.2 - globals: 13.24.0 - graphemer: 1.4.0 - ignore: 5.3.2 - imurmurhash: 0.1.4 - is-glob: 4.0.3 - is-path-inside: 3.0.3 - js-yaml: 4.1.0 - json-stable-stringify-without-jsonify: 1.0.1 - levn: 0.4.1 - lodash.merge: 4.6.2 - minimatch: 3.1.2 - natural-compare: 1.4.0 - optionator: 0.9.4 - strip-ansi: 6.0.1 - text-table: 0.2.0 - transitivePeerDependencies: - - supports-color - dev: false - /esm-env@1.1.4: + esm-env@1.1.4: resolution: {integrity: sha512-oO82nKPHKkzIj/hbtuDYy/JHqBHFlMIW36SDiPCVsj87ntDLcWN+sJ1erdVryd4NxODacFTsdrIE3b7IamqbOg==} - /espree@9.6.1: + espree@9.6.1: resolution: {integrity: sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - dependencies: - acorn: 8.12.1 - acorn-jsx: 5.3.2(acorn@8.12.1) - eslint-visitor-keys: 3.4.3 - dev: false - /esquery@1.6.0: + esprima@4.0.1: + resolution: {integrity: sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==} + engines: {node: '>=4'} + hasBin: true + + esquery@1.6.0: resolution: {integrity: sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==} engines: {node: '>=0.10'} - dependencies: - estraverse: 5.3.0 - dev: false - /esrap@1.2.2: + esrap@1.2.2: resolution: {integrity: sha512-F2pSJklxx1BlQIQgooczXCPHmcWpn6EsP5oo73LQfonG9fIlIENQ8vMmfGXeojP9MrkzUNAfyU5vdFlR9shHAw==} - dependencies: - '@jridgewell/sourcemap-codec': 1.5.0 - '@types/estree': 1.0.5 - /esrecurse@4.3.0: + esrecurse@4.3.0: resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} engines: {node: '>=4.0'} - dependencies: - estraverse: 5.3.0 - dev: false - /estraverse@5.3.0: + estraverse@5.3.0: resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} engines: {node: '>=4.0'} - dev: false - /esutils@2.0.3: + estree-walker@2.0.2: + resolution: {integrity: sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==} + + esutils@2.0.3: resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} engines: {node: '>=0.10.0'} - dev: false - /fast-check@3.23.0: + extendable-error@0.1.7: + resolution: {integrity: sha512-UOiS2in6/Q0FK0R0q6UY9vYpQ21mr/Qn1KOnte7vsACuNJf514WvCCUHSRCPcgjPT2bAhNIJdlE6bVap1GKmeg==} + + external-editor@3.1.0: + resolution: {integrity: sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==} + engines: {node: '>=4'} + + fast-check@3.23.0: resolution: {integrity: sha512-TmpgSeJ2jUV+FNWnSy9iIE9bOV9nCNQ4it+K9BpCNT9JsQOfZYznWGSbMw+Wa4uusEss0IcL/trFVoRxS6IuAA==} engines: {node: '>=8.0.0'} - requiresBuild: true - dependencies: - pure-rand: 6.1.0 - dev: true - optional: true - /fast-deep-equal@3.1.3: + fast-deep-equal@3.1.3: resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} - dev: false - /fast-glob@3.3.2: + fast-glob@3.3.2: resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} engines: {node: '>=8.6.0'} - dependencies: - '@nodelib/fs.stat': 2.0.5 - '@nodelib/fs.walk': 1.2.8 - glob-parent: 5.1.2 - merge2: 1.4.1 - micromatch: 4.0.8 - /fast-json-stable-stringify@2.1.0: + fast-json-stable-stringify@2.1.0: resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} - dev: false - /fast-levenshtein@2.0.6: + fast-levenshtein@2.0.6: resolution: {integrity: sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==} - dev: false - /fastq@1.17.1: + fast-xml-parser@4.4.1: + resolution: {integrity: sha512-xkjOecfnKGkSsOwtZ5Pz7Us/T6mrbPQrq0nh+aCO5V9nk5NLWmasAHumTKjiPJPWANe+kAZ84Jc8ooJkzZ88Sw==} + hasBin: true + + fastq@1.17.1: resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} - dependencies: - reusify: 1.0.4 - /fdir@6.4.2: + fdir@6.4.2: resolution: {integrity: sha512-KnhMXsKSPZlAhp7+IjUkRZKPb4fUyccpDrdFXbi4QL1qkmFh9kVY09Yox+n4MaOb3lHZ1Tv829C3oaaXoMYPDQ==} peerDependencies: picomatch: ^3 || ^4 peerDependenciesMeta: picomatch: optional: true - dev: true - /file-entry-cache@6.0.1: + fetch-blob@3.2.0: + resolution: {integrity: sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==} + engines: {node: ^12.20 || >= 14.13} + + file-entry-cache@6.0.1: resolution: {integrity: sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==} engines: {node: ^10.12.0 || >=12.0.0} - dependencies: - flat-cache: 3.2.0 - dev: false - /fill-range@7.1.1: + filing-cabinet@5.0.2: + resolution: {integrity: sha512-RZlFj8lzyu6jqtFBeXNqUjjNG6xm+gwXue3T70pRxw1W40kJwlgq0PSWAmh0nAnn5DHuBIecLXk9+1VKS9ICXA==} + engines: {node: '>=18'} + hasBin: true + + fill-range@7.1.1: resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} engines: {node: '>=8'} - dependencies: - to-regex-range: 5.0.1 - /find-up@5.0.0: + find-up@4.1.0: + resolution: {integrity: sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==} + engines: {node: '>=8'} + + find-up@5.0.0: resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} engines: {node: '>=10'} - dependencies: - locate-path: 6.0.0 - path-exists: 4.0.0 - dev: false - /flat-cache@3.2.0: + flat-cache@3.2.0: resolution: {integrity: sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==} engines: {node: ^10.12.0 || >=12.0.0} - dependencies: - flatted: 3.3.1 - keyv: 4.5.4 - rimraf: 3.0.2 - dev: false - /flatted@3.3.1: + flatted@3.3.1: resolution: {integrity: sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==} - dev: false - /focus-trap@7.6.0: + focus-trap@7.6.0: resolution: {integrity: sha512-1td0l3pMkWJLFipobUcGaf+5DTY4PLDDrcqoSaKP8ediO/CoWCCYk/fT/Y2A4e6TNB+Sh6clRJCjOPPnKoNHnQ==} - dependencies: - tabbable: 6.2.0 - dev: true - /foreground-child@3.3.0: + foreground-child@3.3.0: resolution: {integrity: sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==} engines: {node: '>=14'} - dependencies: - cross-spawn: 7.0.3 - signal-exit: 4.1.0 - dev: true - /formsnap@1.0.1(svelte@5.1.9)(sveltekit-superforms@2.20.0): + formdata-polyfill@4.0.10: + resolution: {integrity: sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==} + engines: {node: '>=12.20.0'} + + formsnap@1.0.1: resolution: {integrity: sha512-TvU9CoLSiacW1c7wXhLiyVpyy/LBfG0CEFDbs3M3jrsxBSrkTpsuhbQ8JYKY3CNCmIhZlgxCH+Vqr7RBF9G53w==} peerDependencies: svelte: ^4.0.0 || ^5.0.0-next.1 sveltekit-superforms: ^2.3.0 - dependencies: - nanoid: 5.0.8 - svelte: 5.1.9 - sveltekit-superforms: 2.20.0(@sveltejs/kit@2.7.4)(svelte@5.1.9)(typescript@5.5.4) - dev: true - /fraction.js@4.3.7: + fraction.js@4.3.7: resolution: {integrity: sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew==} - dev: true - /fs.realpath@1.0.0: + fs-extra@11.2.0: + resolution: {integrity: sha512-PmDi3uwK5nFuXh7XDTlVnS17xJS7vW36is2+w3xcv8SVxiB4NyATf4ctkVY5bkSjX0Y4nbvZCq1/EjtEyr9ktw==} + engines: {node: '>=14.14'} + + fs-extra@7.0.1: + resolution: {integrity: sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==} + engines: {node: '>=6 <7 || >=8'} + + fs-extra@8.1.0: + resolution: {integrity: sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==} + engines: {node: '>=6 <7 || >=8'} + + fs.realpath@1.0.0: resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} - dev: false - /fsevents@2.3.3: + fsevents@2.3.3: resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} os: [darwin] - requiresBuild: true - dev: true - optional: true - /function-bind@1.1.2: + function-bind@1.1.2: resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - dev: true - /gensync@1.0.0-beta.2: + gensync@1.0.0-beta.2: resolution: {integrity: sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==} engines: {node: '>=6.9.0'} - dev: true - /glob-parent@5.1.2: + get-amd-module-type@6.0.0: + resolution: {integrity: sha512-hFM7oivtlgJ3d6XWD6G47l8Wyh/C6vFw5G24Kk1Tbq85yh5gcM8Fne5/lFhiuxB+RT6+SI7I1ThB9lG4FBh3jw==} + engines: {node: '>=18'} + + get-own-enumerable-property-symbols@3.0.2: + resolution: {integrity: sha512-I0UBV/XOz1XkIJHEUDMZAbzCThU/H8DxmSfmdGcKPnVhu2VfFqr34jr9777IyaTYvxjedWhqVIilEDsCdP5G6g==} + + glob-parent@5.1.2: resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} engines: {node: '>= 6'} - dependencies: - is-glob: 4.0.3 - /glob-parent@6.0.2: + glob-parent@6.0.2: resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==} engines: {node: '>=10.13.0'} - dependencies: - is-glob: 4.0.3 - /glob@10.4.5: + glob@10.4.5: resolution: {integrity: sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==} hasBin: true - dependencies: - foreground-child: 3.3.0 - jackspeak: 3.4.3 - minimatch: 9.0.5 - minipass: 7.1.2 - package-json-from-dist: 1.0.1 - path-scurry: 1.11.1 - dev: true - /glob@7.2.3: + glob@7.2.3: resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} deprecated: Glob versions prior to v9 are no longer supported - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 3.1.2 - once: 1.4.0 - path-is-absolute: 1.0.1 - dev: false - /globals@11.12.0: + globals@11.12.0: resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==} engines: {node: '>=4'} - dev: true - /globals@13.24.0: + globals@13.24.0: resolution: {integrity: sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==} engines: {node: '>=8'} - dependencies: - type-fest: 0.20.2 - dev: false - /globalyzer@0.1.0: + globalyzer@0.1.0: resolution: {integrity: sha512-40oNTM9UfG6aBmuKxk/giHn5nQ8RVz/SS4Ir6zgzOv9/qC3kKZ9v4etGTcJbEl/NyVQH7FGU7d+X1egr57Md2Q==} - dev: true - /globby@11.1.0: + globby@11.1.0: resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} engines: {node: '>=10'} - dependencies: - array-union: 2.1.0 - dir-glob: 3.0.1 - fast-glob: 3.3.2 - ignore: 5.3.2 - merge2: 1.4.1 - slash: 3.0.0 - dev: false - /globrex@0.1.2: + globrex@0.1.2: resolution: {integrity: sha512-uHJgbwAMwNFf5mLst7IWLNg14x1CkeqglJb/K3doi4dw6q2IvAAmM/Y81kevy83wP+Sst+nutFTYOGg3d1lsxg==} - dev: true - /graphemer@1.4.0: + gonzales-pe@4.3.0: + resolution: {integrity: sha512-otgSPpUmdWJ43VXyiNgEYE4luzHCL2pz4wQ0OnDluC6Eg4Ko3Vexy/SrSynglw/eR+OhkzmqFCZa/OFa/RgAOQ==} + engines: {node: '>=0.6.0'} + hasBin: true + + graceful-fs@4.2.11: + resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} + + graphemer@1.4.0: resolution: {integrity: sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==} - dev: false - /has-flag@4.0.0: + has-flag@4.0.0: resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} engines: {node: '>=8'} - dev: false - /hasown@2.0.2: + hasown@2.0.2: resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} engines: {node: '>= 0.4'} - dependencies: - function-bind: 1.1.2 - dev: true - /ignore@5.3.2: + hast-util-to-html@9.0.3: + resolution: {integrity: sha512-M17uBDzMJ9RPCqLMO92gNNUDuBSq10a25SDBI08iCCxmorf4Yy6sYHK57n9WAbRAAaU+DuR4W6GN9K4DFZesYg==} + + hast-util-whitespace@3.0.0: + resolution: {integrity: sha512-88JUN06ipLwsnv+dVn+OIYOvAuvBMy/Qoi6O7mQHxdPXpjy+Cd6xRkWwux7DKO+4sYILtLBRIKgsdpS2gQc7qw==} + + html-void-elements@3.0.0: + resolution: {integrity: sha512-bEqo66MRXsUGxWHV5IP0PUiAWwoEjba4VCzg0LjFJBpchPaTfyfCKTG6bc5F8ucKec3q5y6qOdGyYTSBEvhCrg==} + + https-proxy-agent@7.0.5: + resolution: {integrity: sha512-1e4Wqeblerz+tMKPIq2EMGiiWW1dIjZOksyHWSUm1rmuvw/how9hBHZ38lAGj5ID4Ik6EdkOw7NmWPy6LAwalw==} + engines: {node: '>= 14'} + + human-id@1.0.2: + resolution: {integrity: sha512-UNopramDEhHJD+VR+ehk8rOslwSfByxPIZyJRfV739NDhN5LF1fa1MqnzKm2lGTQRjNrjK19Q5fhkgIfjlVUKw==} + + iconv-lite@0.4.24: + resolution: {integrity: sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==} + engines: {node: '>=0.10.0'} + + ieee754@1.2.1: + resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} + + ignore@5.3.2: resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} engines: {node: '>= 4'} - dev: false - /import-fresh@3.3.0: + import-fresh@3.3.0: resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} engines: {node: '>=6'} - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - dev: false - /import-meta-resolve@4.1.0: + import-meta-resolve@4.1.0: resolution: {integrity: sha512-I6fiaX09Xivtk+THaMfAwnA3MVA5Big1WHF1Dfx9hFuvNIWpXnorlkzhcQf6ehrqQiiZECRt1poOAkPmer3ruw==} - dev: true - /imurmurhash@0.1.4: + imurmurhash@0.1.4: resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} engines: {node: '>=0.8.19'} - dev: false - /inflight@1.0.6: + inflight@1.0.6: resolution: {integrity: sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==} deprecated: This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful. - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - dev: false - /inherits@2.0.4: + inherits@2.0.4: resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - dev: false - /inline-style-parser@0.2.4: + ini@1.3.8: + resolution: {integrity: sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==} + + inline-style-parser@0.2.4: resolution: {integrity: sha512-0aO8FkhNZlj/ZIbNi7Lxxr12obT7cL1moPfE4tg1LkX7LlLfC6DeX4l2ZEud1ukP9jNQyNnfzQVqwbwmAATY4Q==} - dev: true - /is-binary-path@2.1.0: + interpret@1.4.0: + resolution: {integrity: sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==} + engines: {node: '>= 0.10'} + + is-binary-path@2.1.0: resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} engines: {node: '>=8'} - dependencies: - binary-extensions: 2.3.0 - dev: true - /is-core-module@2.15.1: + is-core-module@2.15.1: resolution: {integrity: sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==} engines: {node: '>= 0.4'} - dependencies: - hasown: 2.0.2 - dev: true - /is-extglob@2.1.1: + is-extglob@2.1.1: resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} engines: {node: '>=0.10.0'} - /is-fullwidth-code-point@3.0.0: + is-fullwidth-code-point@3.0.0: resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} engines: {node: '>=8'} - dev: true - /is-glob@4.0.3: + is-glob@4.0.3: resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} engines: {node: '>=0.10.0'} - dependencies: - is-extglob: 2.1.1 - /is-number@7.0.0: + is-interactive@1.0.0: + resolution: {integrity: sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==} + engines: {node: '>=8'} + + is-number@7.0.0: resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} engines: {node: '>=0.12.0'} - /is-path-inside@3.0.3: + is-obj@1.0.1: + resolution: {integrity: sha512-l4RyHgRqGN4Y3+9JHVrNqO+tN0rV5My76uW5/nuO4K1b6vw5G8d/cmFjP9tRfEsdhZNt0IFdZuK/c2Vr4Nb+Qg==} + engines: {node: '>=0.10.0'} + + is-path-inside@3.0.3: resolution: {integrity: sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==} engines: {node: '>=8'} - dev: false - /is-reference@3.0.2: + is-reference@3.0.2: resolution: {integrity: sha512-v3rht/LgVcsdZa3O2Nqs+NMowLOxeOm7Ay9+/ARQ2F+qEoANRcqrjAZKGN0v8ymUetZGgkp26LTnGT7H0Qo9Pg==} - dependencies: - '@types/estree': 1.0.5 - /isexe@2.0.0: + is-regexp@1.0.0: + resolution: {integrity: sha512-7zjFAPO4/gwyQAAgRRmqeEeyIICSdmCqa3tsVHMdBzaXXRiqopZL4Cyghg/XulGWrtABTpbnYYzzIRffLkP4oA==} + engines: {node: '>=0.10.0'} + + is-subdir@1.2.0: + resolution: {integrity: sha512-2AT6j+gXe/1ueqbW6fLZJiIw3F8iXGJtt0yDrZaBhAZEG1raiTxKWU+IPqMCzQAXOUCKdA4UDMgacKH25XG2Cw==} + engines: {node: '>=4'} + + is-unicode-supported@0.1.0: + resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} + engines: {node: '>=10'} + + is-url-superb@4.0.0: + resolution: {integrity: sha512-GI+WjezhPPcbM+tqE9LnmsY5qqjwHzTvjJ36wxYX5ujNXefSUJ/T17r5bqDV8yLhcgB59KTPNOc9O9cmHTPWsA==} + engines: {node: '>=10'} + + is-url@1.2.4: + resolution: {integrity: sha512-ITvGim8FhRiYe4IQ5uHSkj7pVaPDrCTkNd3yq3cV7iZAcJdHTUMPMEHcqSOy9xZ9qFenQCvi+2wjH9a1nXqHww==} + + is-windows@1.0.2: + resolution: {integrity: sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==} + engines: {node: '>=0.10.0'} + + isexe@2.0.0: resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - /jackspeak@3.4.3: + jackspeak@3.4.3: resolution: {integrity: sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==} - dependencies: - '@isaacs/cliui': 8.0.2 - optionalDependencies: - '@pkgjs/parseargs': 0.11.0 - dev: true - /jiti@1.21.6: + jiti@1.21.6: resolution: {integrity: sha512-2yTgeWTWzMWkHu6Jp9NKgePDaYHbntiwvYuuJLbbN9vl7DC9DvXKOB2BC3ZZ92D3cvV/aflH0osDfwpHepQ53w==} hasBin: true - dev: true - /joi@17.13.3: + joi@17.13.3: resolution: {integrity: sha512-otDA4ldcIx+ZXsKHWmp0YizCweVRZG96J10b0FevjfuncLO1oX59THoAmHkNubYJ+9gWsYsp5k8v4ib6oDv1fA==} - requiresBuild: true - dependencies: - '@hapi/hoek': 9.3.0 - '@hapi/topo': 5.1.0 - '@sideway/address': 4.1.5 - '@sideway/formula': 3.0.1 - '@sideway/pinpoint': 2.0.0 - dev: true - optional: true - /js-tokens@4.0.0: + js-tokens@4.0.0: resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - dev: true - /js-yaml@4.1.0: + js-yaml@3.14.1: + resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==} + hasBin: true + + js-yaml@4.1.0: resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} hasBin: true - dependencies: - argparse: 2.0.1 - dev: false - /jsesc@3.0.2: + jsesc@3.0.2: resolution: {integrity: sha512-xKqzzWXDttJuOcawBt4KnKHHIf5oQ/Cxax+0PWFG+DFDgHNAdi+TXECADI+RYiFUMmx8792xsMbbgXj4CwnP4g==} engines: {node: '>=6'} hasBin: true - dev: true - /json-buffer@3.0.1: + json-buffer@3.0.1: resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} - dev: false - /json-schema-to-ts@3.1.1: + json-schema-to-ts@3.1.1: resolution: {integrity: sha512-+DWg8jCJG2TEnpy7kOm/7/AxaYoaRbjVB4LFZLySZlWn8exGs3A4OLJR966cVvU26N7X9TWxl+Jsw7dzAqKT6g==} engines: {node: '>=16'} - requiresBuild: true - dependencies: - '@babel/runtime': 7.26.0 - ts-algebra: 2.0.0 - dev: true - optional: true - /json-schema-traverse@0.4.1: + json-schema-traverse@0.4.1: resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} - dev: false - /json-stable-stringify-without-jsonify@1.0.1: + json-stable-stringify-without-jsonify@1.0.1: resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} - dev: false - /json5@2.2.3: + json5@2.2.3: resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} engines: {node: '>=6'} hasBin: true - dev: true - /just-clone@6.2.0: + jsonfile@4.0.0: + resolution: {integrity: sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==} + + jsonfile@6.1.0: + resolution: {integrity: sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==} + + just-clone@6.2.0: resolution: {integrity: sha512-1IynUYEc/HAwxhi3WDpIpxJbZpMCvvrrmZVqvj9EhpvbH8lls7HhdhiByjL7DkAaWlLIzpC0Xc/VPvy/UxLNjA==} - dev: true - /keyv@4.5.4: + keyv@4.5.4: resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} - dependencies: - json-buffer: 3.0.1 - dev: false - /kleur@4.1.5: + kleur@4.1.5: resolution: {integrity: sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ==} engines: {node: '>=6'} - dev: true - /known-css-properties@0.35.0: + known-css-properties@0.35.0: resolution: {integrity: sha512-a/RAk2BfKk+WFGhhOCAYqSiFLc34k8Mt/6NWRI4joER0EYUzXIcFivjjnoD3+XU1DggLn/tZc3DOAgke7l8a4A==} - dev: false - /levn@0.4.1: + levn@0.4.1: resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} engines: {node: '>= 0.8.0'} - dependencies: - prelude-ls: 1.2.1 - type-check: 0.4.0 - dev: false - /libphonenumber-js@1.11.12: + libphonenumber-js@1.11.12: resolution: {integrity: sha512-QkJn9/D7zZ1ucvT++TQSvZuSA2xAWeUytU+DiEQwbPKLyrDpvbul2AFs1CGbRAPpSCCk47aRAb5DX5mmcayp4g==} - requiresBuild: true - dev: true - optional: true - /lilconfig@2.1.0: + lilconfig@2.1.0: resolution: {integrity: sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ==} engines: {node: '>=10'} - /lilconfig@3.1.2: + lilconfig@3.1.2: resolution: {integrity: sha512-eop+wDAvpItUys0FWkHIKeC9ybYrTGbU41U5K7+bttZZeohvnY7M9dZ5kB21GNWiFT2q1OoPTvncPCgSOVO5ow==} engines: {node: '>=14'} - dev: true - /lines-and-columns@1.2.4: + lines-and-columns@1.2.4: resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - dev: true - /locate-character@3.0.0: + linkify-it@5.0.0: + resolution: {integrity: sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ==} + + locate-character@3.0.0: resolution: {integrity: sha512-SW13ws7BjaeJ6p7Q6CO2nchbYEc3X3J6WrmTTDto7yMPqVSZTUyY5Tjbid+Ab8gLnATtygYtiDIJGQRRn2ZOiA==} - /locate-path@6.0.0: + locate-path@5.0.0: + resolution: {integrity: sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==} + engines: {node: '>=8'} + + locate-path@6.0.0: resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} engines: {node: '>=10'} - dependencies: - p-locate: 5.0.0 - dev: false - /lodash.castarray@4.4.0: + lodash.castarray@4.4.0: resolution: {integrity: sha512-aVx8ztPv7/2ULbArGJ2Y42bG1mEQ5mGjpdvrbJcJFU3TbYybe+QlLS4pst9zV52ymy2in1KpFPiZnAOATxD4+Q==} - dev: true - /lodash.isplainobject@4.0.6: + lodash.isplainobject@4.0.6: resolution: {integrity: sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==} - dev: true - /lodash.merge@4.6.2: + lodash.merge@4.6.2: resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} - /lru-cache@10.4.3: + lodash.startcase@4.4.0: + resolution: {integrity: sha512-+WKqsK294HMSc2jEbNgpHpd0JfIBhp7rEV4aqXWqFr6AlXov+SlcgB1Fv01y2kGe3Gc8nMW7VA0SrGuSkRfIEg==} + + lodash@4.17.21: + resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} + + log-symbols@4.1.0: + resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} + engines: {node: '>=10'} + + lru-cache@10.4.3: resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==} - dev: true - /lru-cache@5.1.1: + lru-cache@4.1.5: + resolution: {integrity: sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==} + + lru-cache@5.1.1: resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} - dependencies: - yallist: 3.1.1 - dev: true - /lucide-svelte@0.416.0(svelte@5.1.9): - resolution: {integrity: sha512-1tEN4VZhUXGmV0UCSDPdUjHgdRVZocZFYB2ufoIFie1ux6kQEiwc64gx8WBUGQY9AoN9CPMlTbGMNb6NzaSMmg==} - peerDependencies: - svelte: ^3 || ^4 || ^5.0.0-next.42 - dependencies: - svelte: 5.1.9 - dev: true - - /lucide-svelte@0.454.0(svelte@5.1.9): + lucide-svelte@0.454.0: resolution: {integrity: sha512-TgW17HI7M8LeFZ3NpaDp1LwPGBGMVjx/x81TtK+AacEQvmJcqetqeJNeBT18NMEJP9+zi/Wt+Zc8mo44K5Uszw==} peerDependencies: svelte: ^3 || ^4 || ^5.0.0-next.42 - dependencies: - svelte: 5.1.9 - dev: true - /magic-string@0.30.12: + lunr@2.3.9: + resolution: {integrity: sha512-zTU3DaZaF3Rt9rhN3uBMGQD3dD2/vFQqnvZCDv4dl5iOzq2IZQqTxu90r4E5J+nP70J3ilqVCrbho2eWaeW8Ow==} + + madge@8.0.0: + resolution: {integrity: sha512-9sSsi3TBPhmkTCIpVQF0SPiChj1L7Rq9kU2KDG1o6v2XH9cCw086MopjVCD+vuoL5v8S77DTbVopTO8OUiQpIw==} + engines: {node: '>=18'} + hasBin: true + peerDependencies: + typescript: ^5.4.4 + peerDependenciesMeta: + typescript: + optional: true + + magic-string@0.30.12: resolution: {integrity: sha512-Ea8I3sQMVXr8JhN4z+H/d8zwo+tYDgHE9+5G4Wnrwhs0gaK9fXTKx0Tw5Xwsd/bCPTTZNRAdpyzvoeORe9LYpw==} - dependencies: - '@jridgewell/sourcemap-codec': 1.5.0 - /memoize-weak@1.0.2: + markdown-it@14.1.0: + resolution: {integrity: sha512-a54IwgWPaeBCAAsv13YgmALOF1elABB08FxO9i+r4VFk5Vl4pKokRPeX8u5TCgSsPi6ec1otfLjdOpVcgbpshg==} + hasBin: true + + mdast-util-to-hast@13.2.0: + resolution: {integrity: sha512-QGYKEuUsYT9ykKBCMOEDLsU5JRObWQusAolFMeko/tYPufNkRffBAQjIE+99jbA87xv6FgmjLtwjh9wBWajwAA==} + + mdurl@2.0.0: + resolution: {integrity: sha512-Lf+9+2r+Tdp5wXDXC4PcIBjTDtq4UKjCPMQhKIuzpJNW0b96kVqSwW0bT7FhRSfmAiFYgP+SCRvdrDozfh0U5w==} + + memoize-weak@1.0.2: resolution: {integrity: sha512-gj39xkrjEw7nCn4nJ1M5ms6+MyMlyiGmttzsqAUsAKn6bYKwuTHh/AO3cKPF8IBrTIYTxb0wWXFs3E//Y8VoWQ==} - dev: true - /merge2@1.4.1: + merge2@1.4.1: resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} engines: {node: '>= 8'} - /micromatch@4.0.8: + micromark-util-character@2.1.0: + resolution: {integrity: sha512-KvOVV+X1yLBfs9dCBSopq/+G1PcgT3lAK07mC4BzXi5E7ahzMAF8oIupDDJ6mievI6F+lAATkbQQlQixJfT3aQ==} + + micromark-util-encode@2.0.0: + resolution: {integrity: sha512-pS+ROfCXAGLWCOc8egcBvT0kf27GoWMqtdarNfDcjb6YLuV5cM3ioG45Ys2qOVqeqSbjaKg72vU+Wby3eddPsA==} + + micromark-util-sanitize-uri@2.0.0: + resolution: {integrity: sha512-WhYv5UEcZrbAtlsnPuChHUAsu/iBPOVaEVsntLBIdpibO0ddy8OzavZz3iL2xVvBZOpolujSliP65Kq0/7KIYw==} + + micromark-util-symbol@2.0.0: + resolution: {integrity: sha512-8JZt9ElZ5kyTnO94muPxIGS8oyElRJaiJO8EzV6ZSyGQ1Is8xwl4Q45qU5UOg+bGH4AikWziz0iN4sFLWs8PGw==} + + micromark-util-types@2.0.0: + resolution: {integrity: sha512-oNh6S2WMHWRZrmutsRmDDfkzKtxF+bc2VxLC9dvtrDIRFln627VsFP6fLMgTryGDljgLPjkrzQSDcPrjPyDJ5w==} + + micromatch@4.0.8: resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} engines: {node: '>=8.6'} - dependencies: - braces: 3.0.3 - picomatch: 2.3.1 - /mini-svg-data-uri@1.4.4: + mimic-fn@2.1.0: + resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==} + engines: {node: '>=6'} + + mini-svg-data-uri@1.4.4: resolution: {integrity: sha512-r9deDe9p5FJUPZAk3A59wGH7Ii9YrjjWw0jmw/liSbHl2CHiyXj6FcDXDu2K3TjVAXqiJdaw3xxwlZZr9E6nHg==} hasBin: true - dev: true - /minimatch@3.1.2: + minimatch@10.0.1: + resolution: {integrity: sha512-ethXTt3SGGR+95gudmqJ1eNhRO7eGEGIgYA9vnPatK4/etz2MEVDno5GMCibdMTuBMyElzIlgxMna3K94XDIDQ==} + engines: {node: 20 || >=22} + + minimatch@3.1.2: resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} - dependencies: - brace-expansion: 1.1.11 - dev: false - /minimatch@9.0.5: + minimatch@9.0.5: resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} engines: {node: '>=16 || 14 >=14.17'} - dependencies: - brace-expansion: 2.0.1 - /minipass@7.1.2: + minimist@1.2.8: + resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} + + minipass@7.1.2: resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} engines: {node: '>=16 || 14 >=14.17'} - dev: true - /mode-watcher@0.4.1(svelte@5.1.9): + minizlib@3.0.1: + resolution: {integrity: sha512-umcy022ILvb5/3Djuu8LWeqUa8D68JaBzlttKeMWen48SjabqS3iY5w/vzeMzMUNhLDifyhbOwKDSznB1vvrwg==} + engines: {node: '>= 18'} + + mkdirp@3.0.1: + resolution: {integrity: sha512-+NsyUUAZDmo6YVHzL/stxSu3t9YS1iljliy3BSDrXJ/dkn1KYdmtZODGGjLcc9XLgVVpH4KshHB8XmZgMhaBXg==} + engines: {node: '>=10'} + hasBin: true + + mode-watcher@0.4.1: resolution: {integrity: sha512-bNC+1NXmwEFZtziCdZSgP7HFQTpqJPcQn9GwwJQGSf6SBF3neEPYV1uRwkYuAQwbsvsXIYtzaqgedDzJ7D1mhg==} peerDependencies: svelte: ^4.0.0 || ^5.0.0-next.1 - dependencies: - svelte: 5.1.9 - dev: true - /mri@1.2.0: + module-definition@6.0.0: + resolution: {integrity: sha512-sEGP5nKEXU7fGSZUML/coJbrO+yQtxcppDAYWRE9ovWsTbFoUHB2qDUx564WUzDaBHXsD46JBbIK5WVTwCyu3w==} + engines: {node: '>=18'} + hasBin: true + + module-lookup-amd@9.0.2: + resolution: {integrity: sha512-p7PzSVEWiW9fHRX9oM+V4aV5B2nCVddVNv4DZ/JB6t9GsXY4E+ZVhPpnwUX7bbJyGeeVZqhS8q/JZ/H77IqPFA==} + engines: {node: '>=18'} + hasBin: true + + mri@1.2.0: resolution: {integrity: sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==} engines: {node: '>=4'} - dev: true - /mrmime@2.0.0: + mrmime@2.0.0: resolution: {integrity: sha512-eu38+hdgojoyq63s+yTpN4XMBdt5l8HhMhc4VKLO9KM5caLIBvUm4thi7fFaxyTmCKeNnXZ5pAlBwCUnhA09uw==} engines: {node: '>=10'} - dev: true - /ms@2.1.3: + ms@2.1.3: resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} - /mz@2.7.0: - resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} - dependencies: - any-promise: 1.3.0 - object-assign: 4.1.1 - thenify-all: 1.6.0 - dev: true + mylas@2.1.13: + resolution: {integrity: sha512-+MrqnJRtxdF+xngFfUUkIMQrUUL0KsxbADUkn23Z/4ibGg192Q+z+CQyiYwvWTsYjJygmMR8+w3ZDa98Zh6ESg==} + engines: {node: '>=12.0.0'} - /nanoid@3.3.7: + mz@2.7.0: + resolution: {integrity: sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==} + + nanoid@3.3.7: resolution: {integrity: sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==} engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} hasBin: true - /nanoid@5.0.8: + nanoid@5.0.8: resolution: {integrity: sha512-TcJPw+9RV9dibz1hHUzlLVy8N4X9TnwirAjrU08Juo6BNKggzVfP2ZJ/3ZUSq15Xl5i85i+Z89XBO90pB2PghQ==} engines: {node: ^18 || >=20} hasBin: true - dev: true - /natural-compare@1.4.0: + natural-compare@1.4.0: resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} - dev: false - /node-releases@2.0.18: + node-domexception@1.0.0: + resolution: {integrity: sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==} + engines: {node: '>=10.5.0'} + + node-fetch@3.3.2: + resolution: {integrity: sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + + node-releases@2.0.18: resolution: {integrity: sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==} - dev: true - /normalize-path@3.0.0: + node-source-walk@7.0.0: + resolution: {integrity: sha512-1uiY543L+N7Og4yswvlm5NCKgPKDEXd9AUR9Jh3gen6oOeBsesr6LqhXom1er3eRzSUcVRWXzhv8tSNrIfGHKw==} + engines: {node: '>=18'} + + normalize-path@3.0.0: resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} engines: {node: '>=0.10.0'} - dev: true - /normalize-range@0.1.2: + normalize-range@0.1.2: resolution: {integrity: sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==} engines: {node: '>=0.10.0'} - dev: true - /normalize-url@8.0.1: + normalize-url@8.0.1: resolution: {integrity: sha512-IO9QvjUMWxPQQhs60oOu10CRkWCiZzSUkzbXGGV9pviYl1fXYcvkzQ5jV9z8Y6un8ARoVRl4EtC6v6jNqbaJ/w==} engines: {node: '>=14.16'} - requiresBuild: true - dev: true - optional: true - /object-assign@4.1.1: + npm-normalize-package-bin@4.0.0: + resolution: {integrity: sha512-TZKxPvItzai9kN9H/TkmCtx/ZN/hvr3vUycjlfmH0ootY9yFBzNOpiXAdIn1Iteqsvk4lQn6B5PTrt+n6h8k/w==} + engines: {node: ^18.17.0 || >=20.5.0} + + object-assign@4.1.1: resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} engines: {node: '>=0.10.0'} - dev: true - /object-hash@3.0.0: + object-hash@3.0.0: resolution: {integrity: sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==} engines: {node: '>= 6'} - dev: true - /once@1.4.0: + once@1.4.0: resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} - dependencies: - wrappy: 1.0.2 - dev: false - /optionator@0.9.4: + onetime@5.1.2: + resolution: {integrity: sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==} + engines: {node: '>=6'} + + oniguruma-to-js@0.4.3: + resolution: {integrity: sha512-X0jWUcAlxORhOqqBREgPMgnshB7ZGYszBNspP+tS9hPD3l13CdaXcHbgImoHUHlrvGx/7AvFEkTRhAGYh+jzjQ==} + + optionator@0.9.4: resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==} engines: {node: '>= 0.8.0'} - dependencies: - deep-is: 0.1.4 - fast-levenshtein: 2.0.6 - levn: 0.4.1 - prelude-ls: 1.2.1 - type-check: 0.4.0 - word-wrap: 1.2.5 - dev: false - /p-limit@3.1.0: + ora@5.4.1: + resolution: {integrity: sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==} + engines: {node: '>=10'} + + os-tmpdir@1.0.2: + resolution: {integrity: sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==} + engines: {node: '>=0.10.0'} + + outdent@0.5.0: + resolution: {integrity: sha512-/jHxFIzoMXdqPzTaCpFzAAWhpkSjZPF4Vsn6jAfNpmbH/ymsmd7Qc6VE9BGn0L6YMj6uwpQLxCECpus4ukKS9Q==} + + p-filter@2.1.0: + resolution: {integrity: sha512-ZBxxZ5sL2HghephhpGAQdoskxplTwr7ICaehZwLIlfL6acuVgZPm8yBNuRAFBGEqtD/hmUeq9eqLg2ys9Xr/yw==} + engines: {node: '>=8'} + + p-limit@2.3.0: + resolution: {integrity: sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==} + engines: {node: '>=6'} + + p-limit@3.1.0: resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} engines: {node: '>=10'} - dependencies: - yocto-queue: 0.1.0 - dev: false - /p-locate@5.0.0: + p-locate@4.1.0: + resolution: {integrity: sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==} + engines: {node: '>=8'} + + p-locate@5.0.0: resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} engines: {node: '>=10'} - dependencies: - p-limit: 3.1.0 - dev: false - /package-json-from-dist@1.0.1: + p-map@2.1.0: + resolution: {integrity: sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw==} + engines: {node: '>=6'} + + p-try@2.2.0: + resolution: {integrity: sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==} + engines: {node: '>=6'} + + package-json-from-dist@1.0.1: resolution: {integrity: sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==} - dev: true - /paneforge@1.0.0-next.1(svelte@5.1.9): + package-manager-detector@0.2.2: + resolution: {integrity: sha512-VgXbyrSNsml4eHWIvxxG/nTL4wgybMTXCV2Un/+yEc3aDKKU6nQBZjbeP3Pl3qm9Qg92X/1ng4ffvCeD/zwHgg==} + + paneforge@1.0.0-next.1: resolution: {integrity: sha512-K28RZ/KLqnZJeuZ7TZNv5C51NrQ3UPggf2sIhIXjTd4brpRZpmCA9C2SL2o4CC0U2Ox75KxUFwEXq2A/iDBuZg==} peerDependencies: svelte: ^5.0.0-next.1 - dependencies: - svelte: 5.1.9 - svelte-toolbelt: 0.4.6(svelte@5.1.9) - dev: true - /parent-module@1.0.1: + parent-module@1.0.1: resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} engines: {node: '>=6'} - dependencies: - callsites: 3.1.0 - dev: false - /path-exists@4.0.0: + parse-ms@2.1.0: + resolution: {integrity: sha512-kHt7kzLoS9VBZfUsiKjv43mr91ea+U05EyKkEtqp7vNbHxmaVuEqN7XxeEVnGrMtYOAxGrDElSi96K7EgO1zCA==} + engines: {node: '>=6'} + + path-exists@4.0.0: resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} engines: {node: '>=8'} - dev: false - /path-is-absolute@1.0.1: + path-is-absolute@1.0.1: resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==} engines: {node: '>=0.10.0'} - dev: false - /path-key@3.1.1: + path-key@3.1.1: resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} engines: {node: '>=8'} - /path-parse@1.0.7: + path-parse@1.0.7: resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} - dev: true - /path-scurry@1.11.1: + path-scurry@1.11.1: resolution: {integrity: sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==} engines: {node: '>=16 || 14 >=14.18'} - dependencies: - lru-cache: 10.4.3 - minipass: 7.1.2 - dev: true - /path-type@4.0.0: + path-type@4.0.0: resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} engines: {node: '>=8'} - dev: false - /picocolors@1.1.0: + picocolors@1.1.0: resolution: {integrity: sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==} - /picomatch@2.3.1: + picocolors@1.1.1: + resolution: {integrity: sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==} + + picomatch@2.3.1: resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} engines: {node: '>=8.6'} - /pify@2.3.0: + picomatch@4.0.2: + resolution: {integrity: sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==} + engines: {node: '>=12'} + + pify@2.3.0: resolution: {integrity: sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==} engines: {node: '>=0.10.0'} - dev: true - /pirates@4.0.6: + pify@4.0.1: + resolution: {integrity: sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==} + engines: {node: '>=6'} + + pirates@4.0.6: resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==} engines: {node: '>= 6'} - dev: true - /postcss-import@15.1.0(postcss@8.4.47): + plimit-lit@1.6.1: + resolution: {integrity: sha512-B7+VDyb8Tl6oMJT9oSO2CW8XC/T4UcJGrwOVoNGwOQsQYhlpfajmrMj5xeejqaASq3V/EqThyOeATEOMuSEXiA==} + engines: {node: '>=12'} + + pluralize@8.0.0: + resolution: {integrity: sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==} + engines: {node: '>=4'} + + postcss-import@15.1.0: resolution: {integrity: sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==} engines: {node: '>=14.0.0'} peerDependencies: postcss: ^8.0.0 - dependencies: - postcss: 8.4.47 - postcss-value-parser: 4.2.0 - read-cache: 1.0.0 - resolve: 1.22.8 - dev: true - /postcss-js@4.0.1(postcss@8.4.47): + postcss-js@4.0.1: resolution: {integrity: sha512-dDLF8pEO191hJMtlHFPRa8xsizHaM82MLfNkUHdUtVEV3tgTp5oj+8qbEqYM57SLfc74KSbw//4SeJma2LRVIw==} engines: {node: ^12 || ^14 || >= 16} peerDependencies: postcss: ^8.4.21 - dependencies: - camelcase-css: 2.0.1 - postcss: 8.4.47 - dev: true - /postcss-load-config@3.1.4(postcss@8.4.47): + postcss-load-config@3.1.4: resolution: {integrity: sha512-6DiM4E7v4coTE4uzA8U//WhtPwyhiim3eyjEMFCnUpzbrkK9wJHgKDT2mR+HbtSrd/NubVaYTOpSpjUl8NQeRg==} engines: {node: '>= 10'} peerDependencies: @@ -2731,13 +3388,8 @@ packages: optional: true ts-node: optional: true - dependencies: - lilconfig: 2.1.0 - postcss: 8.4.47 - yaml: 1.10.2 - dev: false - /postcss-load-config@4.0.2(postcss@8.4.47): + postcss-load-config@4.0.2: resolution: {integrity: sha512-bSVhyJGL00wMVoPUzAVAnbEoWyqRxkjv64tUl427SKnPrENtq6hJwUojroMz2VB+Q1edmi4IfrAPpami5VVgMQ==} engines: {node: '>= 14'} peerDependencies: @@ -2748,83 +3400,62 @@ packages: optional: true ts-node: optional: true - dependencies: - lilconfig: 3.1.2 - postcss: 8.4.47 - yaml: 2.6.0 - dev: true - /postcss-nested@6.2.0(postcss@8.4.47): + postcss-nested@6.2.0: resolution: {integrity: sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ==} engines: {node: '>=12.0'} peerDependencies: postcss: ^8.2.14 - dependencies: - postcss: 8.4.47 - postcss-selector-parser: 6.1.2 - dev: true - /postcss-safe-parser@6.0.0(postcss@8.4.47): + postcss-safe-parser@6.0.0: resolution: {integrity: sha512-FARHN8pwH+WiS2OPCxJI8FuRJpTVnn6ZNFiqAM2aeW2LwTHWWmWgIyKC6cUo0L8aeKiF/14MNvnpls6R2PBeMQ==} engines: {node: '>=12.0'} peerDependencies: postcss: ^8.3.3 - dependencies: - postcss: 8.4.47 - dev: false - /postcss-scss@4.0.9(postcss@8.4.47): + postcss-scss@4.0.9: resolution: {integrity: sha512-AjKOeiwAitL/MXxQW2DliT28EKukvvbEWx3LBmJIRN8KfBGZbRTxNYW0kSqi1COiTZ57nZ9NW06S6ux//N1c9A==} engines: {node: '>=12.0'} peerDependencies: postcss: ^8.4.29 - dependencies: - postcss: 8.4.47 - dev: false - /postcss-selector-parser@6.0.10: + postcss-selector-parser@6.0.10: resolution: {integrity: sha512-IQ7TZdoaqbT+LCpShg46jnZVlhWD2w6iQYAcYXfHARZ7X1t/UGhhceQDs5X0cGqKvYlHNOuv7Oa1xmb0oQuA3w==} engines: {node: '>=4'} - dependencies: - cssesc: 3.0.0 - util-deprecate: 1.0.2 - dev: true - /postcss-selector-parser@6.1.2: + postcss-selector-parser@6.1.2: resolution: {integrity: sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==} engines: {node: '>=4'} - dependencies: - cssesc: 3.0.0 - util-deprecate: 1.0.2 - /postcss-value-parser@4.2.0: + postcss-value-parser@4.2.0: resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==} - dev: true - /postcss@8.4.47: + postcss-values-parser@6.0.2: + resolution: {integrity: sha512-YLJpK0N1brcNJrs9WatuJFtHaV9q5aAOj+S4DI5S7jgHlRfm0PIbDCAFRYMQD5SHq7Fy6xsDhyutgS0QOAs0qw==} + engines: {node: '>=10'} + peerDependencies: + postcss: ^8.2.9 + + postcss@8.4.47: resolution: {integrity: sha512-56rxCq7G/XfB4EkXq9Egn5GCqugWvDFjafDOThIdMBsI15iqPqR5r15TfSr1YPYeEI19YeaXMCbY6u88Y76GLQ==} engines: {node: ^10 || ^12 || >=14} - dependencies: - nanoid: 3.3.7 - picocolors: 1.1.0 - source-map-js: 1.2.1 - /prelude-ls@1.2.1: + precinct@12.1.2: + resolution: {integrity: sha512-x2qVN3oSOp3D05ihCd8XdkIPuEQsyte7PSxzLqiRgktu79S5Dr1I75/S+zAup8/0cwjoiJTQztE9h0/sWp9bJQ==} + engines: {node: '>=18'} + hasBin: true + + prelude-ls@1.2.1: resolution: {integrity: sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==} engines: {node: '>= 0.8.0'} - dev: false - /prettier-plugin-svelte@3.2.7(prettier@3.3.3)(svelte@5.1.9): + prettier-plugin-svelte@3.2.7: resolution: {integrity: sha512-/Dswx/ea0lV34If1eDcG3nulQ63YNr5KPDfMsjbdtpSWOxKKJ7nAc2qlVuYwEvCr4raIuredNoR7K4JCkmTGaQ==} peerDependencies: prettier: ^3.0.0 svelte: ^3.2.0 || ^4.0.0-next.0 || ^5.0.0-next.0 - dependencies: - prettier: 3.3.3 - svelte: 5.1.9 - dev: true - /prettier-plugin-tailwindcss@0.6.8(@ianvs/prettier-plugin-sort-imports@4.3.1)(prettier-plugin-svelte@3.2.7)(prettier@3.3.3): + prettier-plugin-tailwindcss@0.6.8: resolution: {integrity: sha512-dGu3kdm7SXPkiW4nzeWKCl3uoImdd5CTZEJGxyypEPL37Wj0HT2pLqjrvSei1nTeuQfO4PUfjeW5cTUNRLZ4sA==} engines: {node: '>=14.21.3'} peerDependencies: @@ -2878,294 +3509,321 @@ packages: optional: true prettier-plugin-svelte: optional: true - dependencies: - '@ianvs/prettier-plugin-sort-imports': 4.3.1(prettier@3.3.3) - prettier: 3.3.3 - prettier-plugin-svelte: 3.2.7(prettier@3.3.3)(svelte@5.1.9) - dev: true - /prettier@3.3.3: + prettier@2.8.8: + resolution: {integrity: sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==} + engines: {node: '>=10.13.0'} + hasBin: true + + prettier@3.3.3: resolution: {integrity: sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew==} engines: {node: '>=14'} - dev: true + hasBin: true - /property-expr@2.0.6: + pretty-ms@7.0.1: + resolution: {integrity: sha512-973driJZvxiGOQ5ONsFhOF/DtzPMOMtgC11kCpUrPGMTgqp2q/1gwzCquocrN33is0VZ5GFHXZYMM9l6h67v2Q==} + engines: {node: '>=10'} + + proc-log@5.0.0: + resolution: {integrity: sha512-Azwzvl90HaF0aCz1JrDdXQykFakSSNPaPoiZ9fm5qJIMHioDZEi7OAdRwSm6rSoPtY3Qutnm3L7ogmg3dc+wbQ==} + engines: {node: ^18.17.0 || >=20.5.0} + + property-expr@2.0.6: resolution: {integrity: sha512-SVtmxhRE/CGkn3eZY1T6pC8Nln6Fr/lu1mKSgRud0eC73whjGfoAogbn78LkD8aFL0zz3bAFerKSnOl7NlErBA==} - requiresBuild: true - dev: true - optional: true - /punycode@2.3.1: + property-information@6.5.0: + resolution: {integrity: sha512-PgTgs/BlvHxOu8QuEN7wi5A0OmXaBcHpmCSTehcs6Uuu9IkDIEo13Hy7n898RHfrQ49vKCoGeWZSaAK01nwVig==} + + pseudomap@1.0.2: + resolution: {integrity: sha512-b/YwNhb8lk1Zz2+bXXpS/LK9OisiZZ1SNsSLxN1x2OXVEhW2Ckr/7mWE5vrC1ZTiJlD9g19jWszTmJsB+oEpFQ==} + + punycode.js@2.3.1: + resolution: {integrity: sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA==} + engines: {node: '>=6'} + + punycode@2.3.1: resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} engines: {node: '>=6'} - dev: false - /pure-rand@6.1.0: + pure-rand@6.1.0: resolution: {integrity: sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==} - requiresBuild: true - dev: true - optional: true - /queue-microtask@1.2.3: + queue-lit@1.5.2: + resolution: {integrity: sha512-tLc36IOPeMAubu8BkW8YDBV+WyIgKlYU7zUNs0J5Vk9skSZ4JfGlPOqplP0aHdfv7HL0B2Pg6nwiq60Qc6M2Hw==} + engines: {node: '>=12'} + + queue-microtask@1.2.3: resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} - /read-cache@1.0.0: - resolution: {integrity: sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==} - dependencies: - pify: 2.3.0 - dev: true + quote-unquote@1.0.0: + resolution: {integrity: sha512-twwRO/ilhlG/FIgYeKGFqyHhoEhqgnKVkcmqMKi2r524gz3ZbDTcyFt38E9xjJI2vT+KbRNHVbnJ/e0I25Azwg==} - /readdirp@3.6.0: + rc@1.2.8: + resolution: {integrity: sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==} + hasBin: true + + read-cache@1.0.0: + resolution: {integrity: sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==} + + read-cmd-shim@5.0.0: + resolution: {integrity: sha512-SEbJV7tohp3DAAILbEMPXavBjAnMN0tVnh4+9G8ihV4Pq3HYF9h8QNez9zkJ1ILkv9G2BjdzwctznGZXgu/HGw==} + engines: {node: ^18.17.0 || >=20.5.0} + + read-yaml-file@1.1.0: + resolution: {integrity: sha512-VIMnQi/Z4HT2Fxuwg5KrY174U1VdUIASQVWXXyqtNRtxSr9IYkn1rsI6Tb6HsrHCmB7gVpNwX6JxPTHcH6IoTA==} + engines: {node: '>=6'} + + readable-stream@3.6.2: + resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} + engines: {node: '>= 6'} + + readdirp@3.6.0: resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} engines: {node: '>=8.10.0'} - dependencies: - picomatch: 2.3.1 - dev: true - /readdirp@4.0.2: + readdirp@4.0.2: resolution: {integrity: sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==} engines: {node: '>= 14.16.0'} - dev: true - /regenerator-runtime@0.14.1: + rechoir@0.6.2: + resolution: {integrity: sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==} + engines: {node: '>= 0.10'} + + regenerator-runtime@0.14.1: resolution: {integrity: sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==} - requiresBuild: true - dev: true - optional: true - /resolve-from@4.0.0: + regex@4.4.0: + resolution: {integrity: sha512-uCUSuobNVeqUupowbdZub6ggI5/JZkYyJdDogddJr60L764oxC2pMZov1fQ3wM9bdyzUILDG+Sqx6NAKAz9rKQ==} + + requirejs-config-file@4.0.0: + resolution: {integrity: sha512-jnIre8cbWOyvr8a5F2KuqBnY+SDA4NXr/hzEZJG79Mxm2WiFQz2dzhC8ibtPJS7zkmBEl1mxSwp5HhC1W4qpxw==} + engines: {node: '>=10.13.0'} + + requirejs@2.3.7: + resolution: {integrity: sha512-DouTG8T1WanGok6Qjg2SXuCMzszOo0eHeH9hDZ5Y4x8Je+9JB38HdTLT4/VA8OaUhBa0JPVHJ0pyBkM1z+pDsw==} + engines: {node: '>=0.4.0'} + hasBin: true + + resolve-dependency-path@4.0.0: + resolution: {integrity: sha512-hlY1SybBGm5aYN3PC4rp15MzsJLM1w+MEA/4KU3UBPfz4S0lL3FL6mgv7JgaA8a+ZTeEQAiF1a1BuN2nkqiIlg==} + engines: {node: '>=18'} + + resolve-from@4.0.0: resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} engines: {node: '>=4'} - dev: false - /resolve@1.22.8: + resolve-from@5.0.0: + resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} + engines: {node: '>=8'} + + resolve@1.22.8: resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} hasBin: true - dependencies: - is-core-module: 2.15.1 - path-parse: 1.0.7 - supports-preserve-symlinks-flag: 1.0.0 - dev: true - /reusify@1.0.4: + restore-cursor@3.1.0: + resolution: {integrity: sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==} + engines: {node: '>=8'} + + reusify@1.0.4: resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} engines: {iojs: '>=1.0.0', node: '>=0.10.0'} - /rimraf@3.0.2: + rimraf@3.0.2: resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==} deprecated: Rimraf versions prior to v4 are no longer supported hasBin: true - dependencies: - glob: 7.2.3 - dev: false - /rollup@4.24.3: + rimraf@5.0.10: + resolution: {integrity: sha512-l0OE8wL34P4nJH/H2ffoaniAokM2qSmrtXHmlpvYr5AVVX8msAyW0l8NVJFDxlSK4u3Uh/f41cQheDVdnYijwQ==} + hasBin: true + + rollup@4.24.3: resolution: {integrity: sha512-HBW896xR5HGmoksbi3JBDtmVzWiPAYqp7wip50hjQ67JbDz61nyoMPdqu1DvVW9asYb2M65Z20ZHsyJCMqMyDg==} engines: {node: '>=18.0.0', npm: '>=8.0.0'} hasBin: true - dependencies: - '@types/estree': 1.0.6 - optionalDependencies: - '@rollup/rollup-android-arm-eabi': 4.24.3 - '@rollup/rollup-android-arm64': 4.24.3 - '@rollup/rollup-darwin-arm64': 4.24.3 - '@rollup/rollup-darwin-x64': 4.24.3 - '@rollup/rollup-freebsd-arm64': 4.24.3 - '@rollup/rollup-freebsd-x64': 4.24.3 - '@rollup/rollup-linux-arm-gnueabihf': 4.24.3 - '@rollup/rollup-linux-arm-musleabihf': 4.24.3 - '@rollup/rollup-linux-arm64-gnu': 4.24.3 - '@rollup/rollup-linux-arm64-musl': 4.24.3 - '@rollup/rollup-linux-powerpc64le-gnu': 4.24.3 - '@rollup/rollup-linux-riscv64-gnu': 4.24.3 - '@rollup/rollup-linux-s390x-gnu': 4.24.3 - '@rollup/rollup-linux-x64-gnu': 4.24.3 - '@rollup/rollup-linux-x64-musl': 4.24.3 - '@rollup/rollup-win32-arm64-msvc': 4.24.3 - '@rollup/rollup-win32-ia32-msvc': 4.24.3 - '@rollup/rollup-win32-x64-msvc': 4.24.3 - fsevents: 2.3.3 - dev: true - /run-parallel@1.2.0: + run-parallel@1.2.0: resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} - dependencies: - queue-microtask: 1.2.3 - /runed@0.15.3(svelte@5.1.9): + runed@0.15.3: resolution: {integrity: sha512-HtayB+loDcGdqJDHW8JFdsZzGQMyVzim6+s3052MkjZnwmwDstmF+cusMeTssBe6TCdt5i9D/Tb+KYXN3L0kXA==} peerDependencies: svelte: ^5.0.0-next.1 - dependencies: - esm-env: 1.1.4 - svelte: 5.1.9 - dev: true - /sade@1.8.1: + sade@1.8.1: resolution: {integrity: sha512-xal3CZX1Xlo/k4ApwCFrHVACi9fBqJ7V+mwhBsuf/1IOKbBy098Fex+Wa/5QMubw09pSZ/u8EY8PWgevJsXp1A==} engines: {node: '>=6'} - dependencies: - mri: 1.2.0 - dev: true - /semver@6.3.1: + safe-buffer@5.2.1: + resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} + + safer-buffer@2.1.2: + resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} + + sass-lookup@6.0.1: + resolution: {integrity: sha512-nl9Wxbj9RjEJA5SSV0hSDoU2zYGtE+ANaDS4OFUR7nYrquvBFvPKZZtQHe3lvnxCcylEDV00KUijjdMTUElcVQ==} + engines: {node: '>=18'} + hasBin: true + + semver@6.3.1: resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} hasBin: true - dev: true - /semver@7.6.3: + semver@7.6.3: resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==} engines: {node: '>=10'} hasBin: true - /set-cookie-parser@2.7.1: + set-cookie-parser@2.7.1: resolution: {integrity: sha512-IOc8uWeOZgnb3ptbCURJWNjWUPcO3ZnTTdzsurqERrP6nPyv+paC55vJM0LpOlT2ne+Ix+9+CRG1MNLlyZ4GjQ==} - dev: true - /shebang-command@2.0.0: + shebang-command@1.2.0: + resolution: {integrity: sha512-EV3L1+UQWGor21OmnvojK36mhg+TyIKDh3iFBKBohr5xeXIhNBcx8oWdgkTEEQ+BEFFYdLRuqMfd5L84N1V5Vg==} + engines: {node: '>=0.10.0'} + + shebang-command@2.0.0: resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} engines: {node: '>=8'} - dependencies: - shebang-regex: 3.0.0 - /shebang-regex@3.0.0: + shebang-regex@1.0.0: + resolution: {integrity: sha512-wpoSFAxys6b2a2wHZ1XpDSgD7N9iVjg29Ph9uV/uaP9Ex/KXlkTZTeddxDPSYQpgvzKLGJke2UU0AzoGCjNIvQ==} + engines: {node: '>=0.10.0'} + + shebang-regex@3.0.0: resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} engines: {node: '>=8'} - /signal-exit@4.1.0: + shelljs@0.8.5: + resolution: {integrity: sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow==} + engines: {node: '>=4'} + hasBin: true + + shiki@1.22.2: + resolution: {integrity: sha512-3IZau0NdGKXhH2bBlUk4w1IHNxPh6A5B2sUpyY+8utLu2j/h1QpFkAaUA1bAMxOWWGtTWcAh531vnS4NJKS/lA==} + + shx@0.3.4: + resolution: {integrity: sha512-N6A9MLVqjxZYcVn8hLmtneQWIJtp8IKzMP4eMnx+nqkvXoqinUPCbUFLp2UcWTEIUONhlk0ewxr/jaVGlc+J+g==} + engines: {node: '>=6'} + hasBin: true + + signal-exit@3.0.7: + resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} + + signal-exit@4.1.0: resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} engines: {node: '>=14'} - dev: true - /sirv@3.0.0: + sirv@3.0.0: resolution: {integrity: sha512-BPwJGUeDaDCHihkORDchNyyTvWFhcusy1XMmhEVTQTwGeybFbp8YEmB+njbPnth1FibULBSBVwCQni25XlCUDg==} engines: {node: '>=18'} - dependencies: - '@polka/url': 1.0.0-next.28 - mrmime: 2.0.0 - totalist: 3.0.1 - dev: true - /slash@3.0.0: + slash@3.0.0: resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} engines: {node: '>=8'} - dev: false - /source-map-js@1.2.1: + source-map-js@1.2.1: resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} engines: {node: '>=0.10.0'} - /source-map-support@0.5.21: + source-map-support@0.5.21: resolution: {integrity: sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==} - requiresBuild: true - dependencies: - buffer-from: 1.1.2 - source-map: 0.6.1 - dev: true - optional: true - /source-map@0.6.1: + source-map@0.6.1: resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==} engines: {node: '>=0.10.0'} - requiresBuild: true - dev: true - optional: true - /string-width@4.2.3: + space-separated-tokens@2.0.2: + resolution: {integrity: sha512-PEGlAwrG8yXGXRjW32fGbg66JAlOAwbObuqVoJpv/mRgoWDQfgH1wDPvtzWyUSNAXBGSk8h755YDbbcEy3SH2Q==} + + spawndamnit@2.0.0: + resolution: {integrity: sha512-j4JKEcncSjFlqIwU5L/rp2N5SIPsdxaRsIv678+TZxZ0SRDJTm8JrxJMjE/XuiEZNEir3S8l0Fa3Ke339WI4qA==} + + sprintf-js@1.0.3: + resolution: {integrity: sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==} + + stream-to-array@2.3.0: + resolution: {integrity: sha512-UsZtOYEn4tWU2RGLOXr/o/xjRBftZRlG3dEWoaHr8j4GuypJ3isitGbVyjQKAuMu+xbiop8q224TjiZWc4XTZA==} + + string-width@4.2.3: resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} engines: {node: '>=8'} - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - dev: true - /string-width@5.1.2: + string-width@5.1.2: resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} engines: {node: '>=12'} - dependencies: - eastasianwidth: 0.2.0 - emoji-regex: 9.2.2 - strip-ansi: 7.1.0 - dev: true - /strip-ansi@6.0.1: + string_decoder@1.3.0: + resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} + + stringify-entities@4.0.4: + resolution: {integrity: sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg==} + + stringify-object@3.3.0: + resolution: {integrity: sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw==} + engines: {node: '>=4'} + + strip-ansi@6.0.1: resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} engines: {node: '>=8'} - dependencies: - ansi-regex: 5.0.1 - /strip-ansi@7.1.0: + strip-ansi@7.1.0: resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} engines: {node: '>=12'} - dependencies: - ansi-regex: 6.1.0 - dev: true - /strip-json-comments@3.1.1: + strip-bom@3.0.0: + resolution: {integrity: sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==} + engines: {node: '>=4'} + + strip-json-comments@2.0.1: + resolution: {integrity: sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==} + engines: {node: '>=0.10.0'} + + strip-json-comments@3.1.1: resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} engines: {node: '>=8'} - dev: false - /style-to-object@1.0.8: + strnum@1.0.5: + resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} + + style-to-object@1.0.8: resolution: {integrity: sha512-xT47I/Eo0rwJmaXC4oilDGDWLohVhR6o/xAQcPQN8q6QBuZVL8qMYL85kLmST5cPjAorwvqIA4qXTRQoYHaL6g==} - dependencies: - inline-style-parser: 0.2.4 - dev: true - /sucrase@3.35.0: + stylus-lookup@6.0.0: + resolution: {integrity: sha512-RaWKxAvPnIXrdby+UWCr1WRfa+lrPMSJPySte4Q6a+rWyjeJyFOLJxr5GrAVfcMCsfVlCuzTAJ/ysYT8p8do7Q==} + engines: {node: '>=18'} + hasBin: true + + sucrase@3.35.0: resolution: {integrity: sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==} engines: {node: '>=16 || 14 >=14.17'} hasBin: true - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - commander: 4.1.1 - glob: 10.4.5 - lines-and-columns: 1.2.4 - mz: 2.7.0 - pirates: 4.0.6 - ts-interface-checker: 0.1.13 - dev: true - /superstruct@2.0.2: + supabase@1.207.9: + resolution: {integrity: sha512-BJPwsAd2UBIpQawcQV3/xKHEZ8YrrkHYpgibxCZbG+RuxuhTtkHG7zR4I3LylIIEwcKp3hmDKu/hO1m2NT5RXA==} + engines: {npm: '>=8'} + hasBin: true + + superstruct@2.0.2: resolution: {integrity: sha512-uV+TFRZdXsqXTL2pRvujROjdZQ4RAlBUS5BTh9IGm+jTqQntYThciG/qu57Gs69yjnVUSqdxF9YLmSnpupBW9A==} engines: {node: '>=14.0.0'} - requiresBuild: true - dev: true - optional: true - /supports-color@7.2.0: + supports-color@7.2.0: resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} engines: {node: '>=8'} - dependencies: - has-flag: 4.0.0 - dev: false - /supports-preserve-symlinks-flag@1.0.0: + supports-preserve-symlinks-flag@1.0.0: resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} engines: {node: '>= 0.4'} - dev: true - /svelte-check@4.0.5(svelte@5.1.9)(typescript@5.5.4): + svelte-check@4.0.5: resolution: {integrity: sha512-icBTBZ3ibBaywbXUat3cK6hB5Du+Kq9Z8CRuyLmm64XIe2/r+lQcbuBx/IQgsbrC+kT2jQ0weVpZSSRIPwB6jQ==} engines: {node: '>= 18.0.0'} hasBin: true peerDependencies: svelte: ^4.0.0 || ^5.0.0-next.0 typescript: '>=5.0.0' - dependencies: - '@jridgewell/trace-mapping': 0.3.25 - chokidar: 4.0.1 - fdir: 6.4.2 - picocolors: 1.1.0 - sade: 1.8.1 - svelte: 5.1.9 - typescript: 5.5.4 - transitivePeerDependencies: - - picomatch - dev: true - /svelte-eslint-parser@0.43.0(svelte@5.1.9): + svelte-eslint-parser@0.43.0: resolution: {integrity: sha512-GpU52uPKKcVnh8tKN5P4UZpJ/fUDndmq7wfsvoVXsyP+aY0anol7Yqo01fyrlaWGMFfm4av5DyrjlaXdLRJvGA==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} peerDependencies: @@ -3173,381 +3831,325 @@ packages: peerDependenciesMeta: svelte: optional: true - dependencies: - eslint-scope: 7.2.2 - eslint-visitor-keys: 3.4.3 - espree: 9.6.1 - postcss: 8.4.47 - postcss-scss: 4.0.9(postcss@8.4.47) - svelte: 5.1.9 - dev: false - /svelte-persisted-store@0.11.0(svelte@5.1.9): + svelte-persisted-store@0.11.0: resolution: {integrity: sha512-9RgJ5DrawGyyfK22A80cfu8Jose3CV8YjEZKz9Tn94rQ0tWyEmYr+XI+wrVF6wjRbW99JMDSVcFRiM3XzVJj/w==} engines: {node: '>=0.14'} peerDependencies: svelte: ^3.48.0 || ^4.0.0 || ^5.0.0-next.0 - dependencies: - svelte: 5.1.9 - dev: true - /svelte-radix@2.0.1(svelte@5.1.9): + svelte-radix@2.0.1: resolution: {integrity: sha512-YrX44Dj+Rp6YZuPSjdmyd6P8QTkb2NXwySUCZYzjwkP6Cl3dZaTBPPeaSOutP3v3ycQ2XwyNOpyn4p0QcN+uYQ==} engines: {node: '>=18.0.0', npm: '>=7.0.0'} peerDependencies: svelte: ^5.0.0 - dependencies: - svelte: 5.1.9 - dev: true - /svelte-sonner@0.3.28(svelte@5.1.9): + svelte-sonner@0.3.28: resolution: {integrity: sha512-K3AmlySeFifF/cKgsYNv5uXqMVNln0NBAacOYgmkQStLa/UoU0LhfAACU6Gr+YYC8bOCHdVmFNoKuDbMEsppJg==} peerDependencies: svelte: ^3.0.0 || ^4.0.0 || ^5.0.0-next.1 - dependencies: - svelte: 5.1.9 - dev: true - /svelte-toolbelt@0.4.6(svelte@5.1.9): + svelte-toolbelt@0.4.6: resolution: {integrity: sha512-k8OUvXBUifHZcAlWeY/HLg/4J0v5m2iOfOhn8fDmjt4AP8ZluaDh9eBFus9lFiLX6O5l6vKqI1dKL5wy7090NQ==} engines: {node: '>=18', pnpm: '>=8.7.0'} peerDependencies: svelte: ^5.0.0-next.126 - dependencies: - clsx: 2.1.1 - style-to-object: 1.0.8 - svelte: 5.1.9 - dev: true - /svelte@5.1.9: + svelte@5.1.9: resolution: {integrity: sha512-nzq+PPKGS2PoEWDjAcXSrKSbXmmmOAxd6dAz1IhRusUpVkFS6DMELWPyBPGwu6TpO/gsgtFXwX0M4+pAR5gzKw==} engines: {node: '>=18'} - dependencies: - '@ampproject/remapping': 2.3.0 - '@jridgewell/sourcemap-codec': 1.5.0 - '@types/estree': 1.0.6 - acorn: 8.12.1 - acorn-typescript: 1.4.13(acorn@8.12.1) - aria-query: 5.3.2 - axobject-query: 4.1.0 - esm-env: 1.1.4 - esrap: 1.2.2 - is-reference: 3.0.2 - locate-character: 3.0.0 - magic-string: 0.30.12 - zimmerframe: 1.1.2 - /sveltekit-superforms@2.20.0(@sveltejs/kit@2.7.4)(svelte@5.1.9)(typescript@5.5.4): + sveltekit-superforms@2.20.0: resolution: {integrity: sha512-5HyA6THKFBHEmJinZ/klu2/0jYr9ElSaXMYc5EO9ptP3x1wQPWVXYl59sMcaSrIjWUlPpayGxVppCyu+x/o4WA==} peerDependencies: '@sveltejs/kit': 1.x || 2.x svelte: 3.x || 4.x || >=5.0.0-next.51 - dependencies: - '@sveltejs/kit': 2.7.4(@sveltejs/vite-plugin-svelte@4.0.0)(svelte@5.1.9)(vite@5.4.10) - devalue: 5.1.1 - just-clone: 6.2.0 - memoize-weak: 1.0.2 - svelte: 5.1.9 - ts-deepmerge: 7.0.1 - optionalDependencies: - '@effect/schema': 0.75.5(effect@3.10.8) - '@exodus/schemasafe': 1.3.0 - '@gcornut/valibot-json-schema': 0.31.0 - '@sinclair/typebox': 0.32.35 - '@typeschema/class-validator': 0.3.0(class-validator@0.14.1) - '@vinejs/vine': 1.8.0 - arktype: 2.0.0-rc.8 - class-validator: 0.14.1 - effect: 3.10.8 - joi: 17.13.3 - json-schema-to-ts: 3.1.1 - superstruct: 2.0.2 - valibot: 0.41.0(typescript@5.5.4) - yup: 1.4.0 - zod: 3.23.8 - zod-to-json-schema: 3.23.5(zod@3.23.8) - transitivePeerDependencies: - - '@types/json-schema' - - typescript - dev: true - /tabbable@6.2.0: + tabbable@6.2.0: resolution: {integrity: sha512-Cat63mxsVJlzYvN51JmVXIgNoUokrIaT2zLclCXjRd8boZ0004U4KCs/sToJ75C6sdlByWxpYnb5Boif1VSFew==} - dev: true - /tailwind-merge@2.5.4: + tailwind-merge@2.5.4: resolution: {integrity: sha512-0q8cfZHMu9nuYP/b5Shb7Y7Sh1B7Nnl5GqNr1U+n2p6+mybvRtayrQ+0042Z5byvTA8ihjlP8Odo8/VnHbZu4Q==} - dev: true - /tailwind-variants@0.2.1(tailwindcss@3.4.14): + tailwind-variants@0.2.1: resolution: {integrity: sha512-2xmhAf4UIc3PijOUcJPA1LP4AbxhpcHuHM2C26xM0k81r0maAO6uoUSHl3APmvHZcY5cZCY/bYuJdfFa4eGoaw==} engines: {node: '>=16.x', pnpm: '>=7.x'} peerDependencies: tailwindcss: '*' - dependencies: - tailwind-merge: 2.5.4 - tailwindcss: 3.4.14 - dev: true - /tailwindcss-animate@1.0.7(tailwindcss@3.4.14): + tailwindcss-animate@1.0.7: resolution: {integrity: sha512-bl6mpH3T7I3UFxuvDEXLxy/VuFxBk5bbzplh7tXI68mwMokNYd1t9qPBHlnyTwfa4JGC4zP516I1hYYtQ/vspA==} peerDependencies: tailwindcss: '>=3.0.0 || insiders' - dependencies: - tailwindcss: 3.4.14 - dev: true - /tailwindcss@3.4.14: + tailwindcss@3.4.14: resolution: {integrity: sha512-IcSvOcTRcUtQQ7ILQL5quRDg7Xs93PdJEk1ZLbhhvJc7uj/OAhYOnruEiwnGgBvUtaUAJ8/mhSw1o8L2jCiENA==} engines: {node: '>=14.0.0'} hasBin: true - dependencies: - '@alloc/quick-lru': 5.2.0 - arg: 5.0.2 - chokidar: 3.6.0 - didyoumean: 1.2.2 - dlv: 1.1.3 - fast-glob: 3.3.2 - glob-parent: 6.0.2 - is-glob: 4.0.3 - jiti: 1.21.6 - lilconfig: 2.1.0 - micromatch: 4.0.8 - normalize-path: 3.0.0 - object-hash: 3.0.0 - picocolors: 1.1.0 - postcss: 8.4.47 - postcss-import: 15.1.0(postcss@8.4.47) - postcss-js: 4.0.1(postcss@8.4.47) - postcss-load-config: 4.0.2(postcss@8.4.47) - postcss-nested: 6.2.0(postcss@8.4.47) - postcss-selector-parser: 6.1.2 - resolve: 1.22.8 - sucrase: 3.35.0 - transitivePeerDependencies: - - ts-node - dev: true - /text-table@0.2.0: + tapable@2.2.1: + resolution: {integrity: sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==} + engines: {node: '>=6'} + + tar@7.4.3: + resolution: {integrity: sha512-5S7Va8hKfV7W5U6g3aYxXmlPoZVAwUMy9AOKyF2fVuZa2UD3qZjg578OrLRt8PcNN1PleVaL/5/yYATNL0ICUw==} + engines: {node: '>=18'} + + tauri-api-adapter@0.3.8: + resolution: {integrity: sha512-HX6VMCWLzMIhTVuEaKbgLbQXR3YpIrzAgZGPv0Nge7zIn352G5vA/QJxQ9rlpFO4Y8S+Ro7VXR7DpVPMisTHaQ==} + peerDependencies: + typescript: ^5.0.0 + + tauri-plugin-clipboard-api@2.1.11: + resolution: {integrity: sha512-VNkGaVPPfRoHg7/rJBcWqsvLvn4/kNEOOlzqwyI9Qdf6g54B3mc31GLZdnq/HWtX0vZskw3J8b/EF9YkASDCBQ==} + + tauri-plugin-network-api@2.0.4: + resolution: {integrity: sha512-CJWF2g+uQifcIlE/AXUnezVjjbyY0FDBxoz4P6BmjNRR/qubpNMfdUnKLqdjX98o5MIXGW+UnyZTfbJo998dFw==} + + tauri-plugin-shellx-api@2.0.11: + resolution: {integrity: sha512-+FKIP1FBHdIQ6tASohww3MOf/8CDvYMYpPg9glO59h8TGVxTNP2ofiOEKLYk8M/o2H4tP7mxxca11QpDAT2LXw==} + + tauri-plugin-system-info-api@2.0.8: + resolution: {integrity: sha512-EFdLXNGp6Zu9SNsZCkU+55A8027OnrVw/TQrd0oJHgfZzs4qvm1iMmSvyid4MLftt33iZDhjCzxYijaaOxeKSg==} + + term-size@2.2.1: + resolution: {integrity: sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==} + engines: {node: '>=8'} + + terser@5.36.0: + resolution: {integrity: sha512-IYV9eNMuFAV4THUspIRXkLakHnV6XO7FEdtKjf/mDyrnqUg9LnlOn6/RwRvM9SZjR4GUq8Nk8zj67FzVARr74w==} + engines: {node: '>=10'} + hasBin: true + + text-table@0.2.0: resolution: {integrity: sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==} - dev: false - /thenify-all@1.6.0: + thenify-all@1.6.0: resolution: {integrity: sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==} engines: {node: '>=0.8'} - dependencies: - thenify: 3.3.1 - dev: true - /thenify@3.3.1: + thenify@3.3.1: resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} - dependencies: - any-promise: 1.3.0 - dev: true - /tiny-case@1.0.3: + tiny-case@1.0.3: resolution: {integrity: sha512-Eet/eeMhkO6TX8mnUteS9zgPbUMQa4I6Kkp5ORiBD5476/m+PIRiumP5tmh5ioJpH7k51Kehawy2UDfsnxxY8Q==} - requiresBuild: true - dev: true - optional: true - /tiny-glob@0.2.9: + tiny-glob@0.2.9: resolution: {integrity: sha512-g/55ssRPUjShh+xkfx9UPDXqhckHEsHr4Vd9zX55oSdGZc/MD0m3sferOkwWtp98bv+kcVfEHtRJgBVJzelrzg==} - dependencies: - globalyzer: 0.1.0 - globrex: 0.1.2 - dev: true - /to-regex-range@5.0.1: + tmp@0.0.33: + resolution: {integrity: sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==} + engines: {node: '>=0.6.0'} + + to-regex-range@5.0.1: resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} engines: {node: '>=8.0'} - dependencies: - is-number: 7.0.0 - /toposort@2.0.2: + toposort@2.0.2: resolution: {integrity: sha512-0a5EOkAUp8D4moMi2W8ZF8jcga7BgZd91O/yabJCFY8az+XSzeGyTKs0Aoo897iV1Nj6guFq8orWDS96z91oGg==} - requiresBuild: true - dev: true - optional: true - /totalist@3.0.1: + totalist@3.0.1: resolution: {integrity: sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==} engines: {node: '>=6'} - dev: true - /ts-algebra@2.0.0: + tr46@0.0.3: + resolution: {integrity: sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==} + + trim-lines@3.0.1: + resolution: {integrity: sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg==} + + ts-algebra@2.0.0: resolution: {integrity: sha512-FPAhNPFMrkwz76P7cdjdmiShwMynZYN6SgOujD1urY4oNm80Ou9oMdmbR45LotcKOXoy7wSmHkRFE6Mxbrhefw==} - requiresBuild: true - dev: true - optional: true - /ts-api-utils@1.4.0(typescript@5.5.4): + ts-api-utils@1.4.0: resolution: {integrity: sha512-032cPxaEKwM+GT3vA5JXNzIaizx388rhsSW79vGRNGXfRRAdEAn2mvk36PvK5HnOchyWZ7afLEXqYCvPCrzuzQ==} engines: {node: '>=16'} peerDependencies: typescript: '>=4.2.0' - dependencies: - typescript: 5.5.4 - dev: false - /ts-deepmerge@7.0.1: + ts-deepmerge@7.0.1: resolution: {integrity: sha512-JBFCmNenZdUCc+TRNCtXVM6N8y/nDQHAcpj5BlwXG/gnogjam1NunulB9ia68mnqYI446giMfpqeBFFkOleh+g==} engines: {node: '>=14.13.1'} - dev: true - /ts-interface-checker@0.1.13: + ts-graphviz@2.1.4: + resolution: {integrity: sha512-0g465/ES70H0h5rcLUqaenKqNYekQaR9W0m0xUGy3FxueGujpGr+0GN2YWlgLIYSE2Xg0W7Uq1Qqnn7Cg+Af2w==} + engines: {node: '>=18'} + + ts-interface-checker@0.1.13: resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} - dev: true - /tslib@2.4.0: + tsc-alias@1.8.10: + resolution: {integrity: sha512-Ibv4KAWfFkFdKJxnWfVtdOmB0Zi1RJVxcbPGiCDsFpCQSsmpWyuzHG3rQyI5YkobWwxFPEyQfu1hdo4qLG2zPw==} + hasBin: true + + tsconfig-paths@4.2.0: + resolution: {integrity: sha512-NoZ4roiN7LnbKn9QqE1amc9DJfzvZXxF4xDavcOWt1BPkdx+m+0gJuPM+S0vCe7zTJMYUP0R8pO2XMr+Y8oLIg==} + engines: {node: '>=6'} + + tslib@2.4.0: resolution: {integrity: sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==} - requiresBuild: true - dev: true - optional: true - /tslib@2.8.1: + tslib@2.8.1: resolution: {integrity: sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==} - dev: true - /turbo-darwin-64@2.2.3: + turbo-darwin-64@2.2.3: resolution: {integrity: sha512-Rcm10CuMKQGcdIBS3R/9PMeuYnv6beYIHqfZFeKWVYEWH69sauj4INs83zKMTUiZJ3/hWGZ4jet9AOwhsssLyg==} cpu: [x64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /turbo-darwin-arm64@2.2.3: + turbo-darwin-arm64@2.2.3: resolution: {integrity: sha512-+EIMHkuLFqUdJYsA3roj66t9+9IciCajgj+DVek+QezEdOJKcRxlvDOS2BUaeN8kEzVSsNiAGnoysFWYw4K0HA==} cpu: [arm64] os: [darwin] - requiresBuild: true - dev: true - optional: true - /turbo-linux-64@2.2.3: + turbo-linux-64@2.2.3: resolution: {integrity: sha512-UBhJCYnqtaeOBQLmLo8BAisWbc9v9daL9G8upLR+XGj6vuN/Nz6qUAhverN4Pyej1g4Nt1BhROnj6GLOPYyqxQ==} cpu: [x64] os: [linux] - requiresBuild: true - dev: true - optional: true - /turbo-linux-arm64@2.2.3: + turbo-linux-arm64@2.2.3: resolution: {integrity: sha512-hJYT9dN06XCQ3jBka/EWvvAETnHRs3xuO/rb5bESmDfG+d9yQjeTMlhRXKrr4eyIMt6cLDt1LBfyi+6CQ+VAwQ==} cpu: [arm64] os: [linux] - requiresBuild: true - dev: true - optional: true - /turbo-windows-64@2.2.3: + turbo-windows-64@2.2.3: resolution: {integrity: sha512-NPrjacrZypMBF31b4HE4ROg4P3nhMBPHKS5WTpMwf7wydZ8uvdEHpESVNMOtqhlp857zbnKYgP+yJF30H3N2dQ==} cpu: [x64] os: [win32] - requiresBuild: true - dev: true - optional: true - /turbo-windows-arm64@2.2.3: + turbo-windows-arm64@2.2.3: resolution: {integrity: sha512-fnNrYBCqn6zgKPKLHu4sOkihBI/+0oYFr075duRxqUZ+1aLWTAGfHZLgjVeLh3zR37CVzuerGIPWAEkNhkWEIw==} cpu: [arm64] os: [win32] - requiresBuild: true - dev: true - optional: true - /turbo@2.2.3: + turbo@2.2.3: resolution: {integrity: sha512-5lDvSqIxCYJ/BAd6rQGK/AzFRhBkbu4JHVMLmGh/hCb7U3CqSnr5Tjwfy9vc+/5wG2DJ6wttgAaA7MoCgvBKZQ==} hasBin: true - optionalDependencies: - turbo-darwin-64: 2.2.3 - turbo-darwin-arm64: 2.2.3 - turbo-linux-64: 2.2.3 - turbo-linux-arm64: 2.2.3 - turbo-windows-64: 2.2.3 - turbo-windows-arm64: 2.2.3 - dev: true - /type-check@0.4.0: + type-check@0.4.0: resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==} engines: {node: '>= 0.8.0'} - dependencies: - prelude-ls: 1.2.1 - dev: false - /type-fest@0.20.2: + type-fest@0.20.2: resolution: {integrity: sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==} engines: {node: '>=10'} - dev: false - /type-fest@2.19.0: + type-fest@2.19.0: resolution: {integrity: sha512-RAH822pAdBgcNMAfWnCBU3CFZcfZ/i1eZjwFU/dsLKumyuuP3niueg2UAukXYF0E2AAoc82ZSSf9J0WQBinzHA==} engines: {node: '>=12.20'} - requiresBuild: true - dev: true - optional: true - /typescript@5.5.4: + typedoc@0.26.11: + resolution: {integrity: sha512-sFEgRRtrcDl2FxVP58Ze++ZK2UQAEvtvvH8rRlig1Ja3o7dDaMHmaBfvJmdGnNEFaLTpQsN8dpvZaTqJSu/Ugw==} + engines: {node: '>= 18'} + hasBin: true + peerDependencies: + typescript: 4.6.x || 4.7.x || 4.8.x || 4.9.x || 5.0.x || 5.1.x || 5.2.x || 5.3.x || 5.4.x || 5.5.x || 5.6.x + + typescript@5.5.4: resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} engines: {node: '>=14.17'} + hasBin: true - /update-browserslist-db@1.1.1(browserslist@4.24.2): + typescript@5.6.3: + resolution: {integrity: sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw==} + engines: {node: '>=14.17'} + hasBin: true + + uc.micro@2.1.0: + resolution: {integrity: sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A==} + + undici-types@5.26.5: + resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==} + + undici-types@6.19.8: + resolution: {integrity: sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==} + + unist-util-is@6.0.0: + resolution: {integrity: sha512-2qCTHimwdxLfz+YzdGfkqNlH0tLi9xjTnHddPmJwtIG9MGsdbutfTc4P+haPD7l7Cjxf/WZj+we5qfVPvvxfYw==} + + unist-util-position@5.0.0: + resolution: {integrity: sha512-fucsC7HjXvkB5R3kTCO7kUjRdrS0BJt3M/FPxmHMBOm8JQi2BsHAHFsy27E0EolP8rp0NzXsJ+jNPyDWvOJZPA==} + + unist-util-stringify-position@4.0.0: + resolution: {integrity: sha512-0ASV06AAoKCDkS2+xw5RXJywruurpbC4JZSm7nr7MOt1ojAzvyyaO+UxZf18j8FCF6kmzCZKcAgN/yu2gm2XgQ==} + + unist-util-visit-parents@6.0.1: + resolution: {integrity: sha512-L/PqWzfTP9lzzEa6CKs0k2nARxTdZduw3zyh8d2NVBnsyvHjSX4TWse388YrrQKbvI8w20fGjGlhgT96WwKykw==} + + unist-util-visit@5.0.0: + resolution: {integrity: sha512-MR04uvD+07cwl/yhVuVWAtw+3GOR/knlL55Nd/wAdblk27GCVt3lqpTivy/tkJcZoNPzTwS1Y+KMojlLDhoTzg==} + + universalify@0.1.2: + resolution: {integrity: sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==} + engines: {node: '>= 4.0.0'} + + universalify@2.0.1: + resolution: {integrity: sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==} + engines: {node: '>= 10.0.0'} + + update-browserslist-db@1.1.1: resolution: {integrity: sha512-R8UzCaa9Az+38REPiJ1tXlImTJXlVfgHZsglwBD/k6nj76ctsH1E3q4doGrukiLQd3sGQYu56r5+lo5r94l29A==} hasBin: true peerDependencies: browserslist: '>= 4.21.0' - dependencies: - browserslist: 4.24.2 - escalade: 3.2.0 - picocolors: 1.1.0 - dev: true - /uri-js@4.4.1: + uri-js@4.4.1: resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} - dependencies: - punycode: 2.3.1 - dev: false - /util-deprecate@1.0.2: + util-deprecate@1.0.2: resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - /valibot@0.31.1: - resolution: {integrity: sha512-2YYIhPrnVSz/gfT2/iXVTrSj92HwchCt9Cga/6hX4B26iCz9zkIsGTS0HjDYTZfTi1Un0X6aRvhBi1cfqs/i0Q==} - requiresBuild: true - dev: true - optional: true + uuid@11.0.2: + resolution: {integrity: sha512-14FfcOJmqdjbBPdDjFQyk/SdT4NySW4eM0zcG+HqbHP5jzuH56xO3J1DGhgs/cEMCfwYi3HQI1gnTO62iaG+tQ==} + hasBin: true - /valibot@0.41.0(typescript@5.5.4): - resolution: {integrity: sha512-igDBb8CTYr8YTQlOKgaN9nSS0Be7z+WRuaeYqGf3Cjz3aKmSnqEmYnkfVjzIuumGqfHpa3fLIvMEAfhrpqN8ng==} - requiresBuild: true + uuid@9.0.1: + resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} + hasBin: true + + valibot@0.31.1: + resolution: {integrity: sha512-2YYIhPrnVSz/gfT2/iXVTrSj92HwchCt9Cga/6hX4B26iCz9zkIsGTS0HjDYTZfTi1Un0X6aRvhBi1cfqs/i0Q==} + + valibot@0.40.0: + resolution: {integrity: sha512-XHKnaVtwHqxPwnGOsLrwka9CEaL7yNeLNp707OKv/bmT29GnPVdl6PxBOZ6BW7hF66/6QT6iVbOlnW7qVPmoKw==} peerDependencies: typescript: '>=5' peerDependenciesMeta: typescript: optional: true - dependencies: - typescript: 5.5.4 - dev: true - optional: true - /validator@13.12.0: + valibot@0.41.0: + resolution: {integrity: sha512-igDBb8CTYr8YTQlOKgaN9nSS0Be7z+WRuaeYqGf3Cjz3aKmSnqEmYnkfVjzIuumGqfHpa3fLIvMEAfhrpqN8ng==} + peerDependencies: + typescript: '>=5' + peerDependenciesMeta: + typescript: + optional: true + + valibot@0.42.1: + resolution: {integrity: sha512-3keXV29Ar5b//Hqi4MbSdV7lfVp6zuYLZuA9V1PvQUsXqogr+u5lvLPLk3A4f74VUXDnf/JfWMN6sB+koJ/FFw==} + peerDependencies: + typescript: '>=5' + peerDependenciesMeta: + typescript: + optional: true + + validator@13.12.0: resolution: {integrity: sha512-c1Q0mCiPlgdTVVVIJIrBuxNicYE+t/7oKeI9MWLj3fh/uq2Pxh/3eeWbVZ4OcGW1TUf53At0njHw5SMdA3tmMg==} engines: {node: '>= 0.10'} - requiresBuild: true - dev: true - optional: true - /vaul-svelte@0.3.2(svelte@5.1.9): + vaul-svelte@0.3.2: resolution: {integrity: sha512-X4OGWttSTVUl417qGDsSFgOvIx24DoiMRY/jaP9z0v9FL8LQQJ0RQ1ZM0QpdyQPRlNd24ewjNQHh5EgYDtfNpw==} peerDependencies: svelte: ^4.0.0 || ^5.0.0-next.1 - dependencies: - bits-ui: 0.21.16(svelte@5.1.9) - svelte: 5.1.9 - dev: true - /vite@5.4.10: + vfile-message@4.0.2: + resolution: {integrity: sha512-jRDZ1IMLttGj41KcZvlrYAaI3CfqpLpfpf+Mfig13viT6NKvRzWZ+lXz0Y5D60w6uJIBAOGq9mSHf0gktF0duw==} + + vfile@6.0.3: + resolution: {integrity: sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q==} + + vite@5.4.10: resolution: {integrity: sha512-1hvaPshuPUtxeQ0hsVH3Mud0ZanOLwVTneA1EgbAM5LhaZEqyPWGRQ7BtaMvUrTDeEaC8pxtj6a6jku3x4z6SQ==} engines: {node: ^18.0.0 || >=20.0.0} hasBin: true @@ -3577,105 +4179,4372 @@ packages: optional: true terser: optional: true - dependencies: - esbuild: 0.21.5 - postcss: 8.4.47 - rollup: 4.24.3 - optionalDependencies: - fsevents: 2.3.3 - dev: true - /vitefu@1.0.3(vite@5.4.10): + vitefu@1.0.3: resolution: {integrity: sha512-iKKfOMBHob2WxEJbqbJjHAkmYgvFDPhuqrO82om83S8RLk+17FtyMBfcyeH8GqD0ihShtkMW/zzJgiA51hCNCQ==} peerDependencies: vite: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0-beta.0 peerDependenciesMeta: vite: optional: true - dependencies: - vite: 5.4.10 - dev: true - /which@2.0.2: + walkdir@0.4.1: + resolution: {integrity: sha512-3eBwRyEln6E1MSzcxcVpQIhRG8Q1jLvEqRmCZqS3dsfXEDR/AhOF4d+jHg1qvDCpYaVRZjENPQyrVxAkQqxPgQ==} + engines: {node: '>=6.0.0'} + + wcwidth@1.0.1: + resolution: {integrity: sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==} + + web-streams-polyfill@3.3.3: + resolution: {integrity: sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==} + engines: {node: '>= 8'} + + webidl-conversions@3.0.1: + resolution: {integrity: sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==} + + whatwg-url@5.0.0: + resolution: {integrity: sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==} + + which@1.3.1: + resolution: {integrity: sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==} + hasBin: true + + which@2.0.2: resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} engines: {node: '>= 8'} hasBin: true + + word-wrap@1.2.5: + resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==} + engines: {node: '>=0.10.0'} + + wrap-ansi@7.0.0: + resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} + engines: {node: '>=10'} + + wrap-ansi@8.1.0: + resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} + engines: {node: '>=12'} + + wrappy@1.0.2: + resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} + + write-file-atomic@6.0.0: + resolution: {integrity: sha512-GmqrO8WJ1NuzJ2DrziEI2o57jKAVIQNf8a18W3nCYU3H7PNWqCCVTeH6/NQE93CIllIgQS98rrmVkYgTX9fFJQ==} + engines: {node: ^18.17.0 || >=20.5.0} + + ws@8.18.0: + resolution: {integrity: sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==} + engines: {node: '>=10.0.0'} + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: '>=5.0.2' + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true + + yallist@2.1.2: + resolution: {integrity: sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==} + + yallist@3.1.1: + resolution: {integrity: sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==} + + yallist@5.0.0: + resolution: {integrity: sha512-YgvUTfwqyc7UXVMrB+SImsVYSmTS8X/tSrtdNZMImM+n7+QTriRXyXim0mBrTXNeqzVF0KWGgHPeiyViFFrNDw==} + engines: {node: '>=18'} + + yaml@1.10.2: + resolution: {integrity: sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==} + engines: {node: '>= 6'} + + yaml@2.6.0: + resolution: {integrity: sha512-a6ae//JvKDEra2kdi1qzCyrJW/WZCgFi8ydDV+eXExl95t+5R+ijnqHJbz9tmMh8FUjx3iv2fCQ4dclAQlO2UQ==} + engines: {node: '>= 14'} + hasBin: true + + yocto-queue@0.1.0: + resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} + engines: {node: '>=10'} + + yup@1.4.0: + resolution: {integrity: sha512-wPbgkJRCqIf+OHyiTBQoJiP5PFuAXaWiJK6AmYkzQAh5/c2K9hzSApBZG5wV9KoKSePF7sAxmNSvh/13YHkFDg==} + + zimmerframe@1.1.2: + resolution: {integrity: sha512-rAbqEGa8ovJy4pyBxZM70hg4pE6gDgaQ0Sl9M3enG3I0d6H4XSAM3GeNGLKnsBpuijUow064sf7ww1nutC5/3w==} + + zod-to-json-schema@3.23.5: + resolution: {integrity: sha512-5wlSS0bXfF/BrL4jPAbz9da5hDlDptdEppYfe+x4eIJ7jioqKG9uUxOwPzqof09u/XeVdrgFu29lZi+8XNDJtA==} + peerDependencies: + zod: ^3.23.3 + + zod@3.23.8: + resolution: {integrity: sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==} + + zwitch@2.0.4: + resolution: {integrity: sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A==} + +snapshots: + + '@alloc/quick-lru@5.2.0': {} + + '@ampproject/remapping@2.3.0': + dependencies: + '@jridgewell/gen-mapping': 0.3.5 + '@jridgewell/trace-mapping': 0.3.25 + + '@ark/schema@0.10.0': + dependencies: + '@ark/util': 0.10.0 + optional: true + + '@ark/util@0.10.0': + optional: true + + '@aws-crypto/crc32@5.2.0': + dependencies: + '@aws-crypto/util': 5.2.0 + '@aws-sdk/types': 3.679.0 + tslib: 2.8.1 + + '@aws-crypto/crc32c@5.2.0': + dependencies: + '@aws-crypto/util': 5.2.0 + '@aws-sdk/types': 3.679.0 + tslib: 2.8.1 + + '@aws-crypto/sha1-browser@5.2.0': + dependencies: + '@aws-crypto/supports-web-crypto': 5.2.0 + '@aws-crypto/util': 5.2.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-locate-window': 3.679.0 + '@smithy/util-utf8': 2.3.0 + tslib: 2.8.1 + + '@aws-crypto/sha256-browser@5.2.0': + dependencies: + '@aws-crypto/sha256-js': 5.2.0 + '@aws-crypto/supports-web-crypto': 5.2.0 + '@aws-crypto/util': 5.2.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-locate-window': 3.679.0 + '@smithy/util-utf8': 2.3.0 + tslib: 2.8.1 + + '@aws-crypto/sha256-js@5.2.0': + dependencies: + '@aws-crypto/util': 5.2.0 + '@aws-sdk/types': 3.679.0 + tslib: 2.8.1 + + '@aws-crypto/supports-web-crypto@5.2.0': + dependencies: + tslib: 2.8.1 + + '@aws-crypto/util@5.2.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/util-utf8': 2.3.0 + tslib: 2.8.1 + + '@aws-sdk/client-s3@3.685.0': + dependencies: + '@aws-crypto/sha1-browser': 5.2.0 + '@aws-crypto/sha256-browser': 5.2.0 + '@aws-crypto/sha256-js': 5.2.0 + '@aws-sdk/client-sso-oidc': 3.682.0(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/client-sts': 3.682.0 + '@aws-sdk/core': 3.679.0 + '@aws-sdk/credential-provider-node': 3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/middleware-bucket-endpoint': 3.679.0 + '@aws-sdk/middleware-expect-continue': 3.679.0 + '@aws-sdk/middleware-flexible-checksums': 3.682.0 + '@aws-sdk/middleware-host-header': 3.679.0 + '@aws-sdk/middleware-location-constraint': 3.679.0 + '@aws-sdk/middleware-logger': 3.679.0 + '@aws-sdk/middleware-recursion-detection': 3.679.0 + '@aws-sdk/middleware-sdk-s3': 3.685.0 + '@aws-sdk/middleware-ssec': 3.679.0 + '@aws-sdk/middleware-user-agent': 3.682.0 + '@aws-sdk/region-config-resolver': 3.679.0 + '@aws-sdk/signature-v4-multi-region': 3.685.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-endpoints': 3.679.0 + '@aws-sdk/util-user-agent-browser': 3.679.0 + '@aws-sdk/util-user-agent-node': 3.682.0 + '@aws-sdk/xml-builder': 3.679.0 + '@smithy/config-resolver': 3.0.10 + '@smithy/core': 2.5.1 + '@smithy/eventstream-serde-browser': 3.0.11 + '@smithy/eventstream-serde-config-resolver': 3.0.8 + '@smithy/eventstream-serde-node': 3.0.10 + '@smithy/fetch-http-handler': 3.2.9 + '@smithy/hash-blob-browser': 3.1.7 + '@smithy/hash-node': 3.0.8 + '@smithy/hash-stream-node': 3.1.7 + '@smithy/invalid-dependency': 3.0.8 + '@smithy/md5-js': 3.0.8 + '@smithy/middleware-content-length': 3.0.10 + '@smithy/middleware-endpoint': 3.2.1 + '@smithy/middleware-retry': 3.0.25 + '@smithy/middleware-serde': 3.0.8 + '@smithy/middleware-stack': 3.0.8 + '@smithy/node-config-provider': 3.1.9 + '@smithy/node-http-handler': 3.2.5 + '@smithy/protocol-http': 4.1.5 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/url-parser': 3.0.8 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.25 + '@smithy/util-defaults-mode-node': 3.0.25 + '@smithy/util-endpoints': 2.1.4 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-retry': 3.0.8 + '@smithy/util-stream': 3.2.1 + '@smithy/util-utf8': 3.0.0 + '@smithy/util-waiter': 3.1.7 + tslib: 2.8.1 + transitivePeerDependencies: + - aws-crt + + '@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0)': + dependencies: + '@aws-crypto/sha256-browser': 5.2.0 + '@aws-crypto/sha256-js': 5.2.0 + '@aws-sdk/client-sts': 3.682.0 + '@aws-sdk/core': 3.679.0 + '@aws-sdk/credential-provider-node': 3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/middleware-host-header': 3.679.0 + '@aws-sdk/middleware-logger': 3.679.0 + '@aws-sdk/middleware-recursion-detection': 3.679.0 + '@aws-sdk/middleware-user-agent': 3.682.0 + '@aws-sdk/region-config-resolver': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-endpoints': 3.679.0 + '@aws-sdk/util-user-agent-browser': 3.679.0 + '@aws-sdk/util-user-agent-node': 3.682.0 + '@smithy/config-resolver': 3.0.10 + '@smithy/core': 2.5.1 + '@smithy/fetch-http-handler': 3.2.9 + '@smithy/hash-node': 3.0.8 + '@smithy/invalid-dependency': 3.0.8 + '@smithy/middleware-content-length': 3.0.10 + '@smithy/middleware-endpoint': 3.2.1 + '@smithy/middleware-retry': 3.0.25 + '@smithy/middleware-serde': 3.0.8 + '@smithy/middleware-stack': 3.0.8 + '@smithy/node-config-provider': 3.1.9 + '@smithy/node-http-handler': 3.2.5 + '@smithy/protocol-http': 4.1.5 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/url-parser': 3.0.8 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.25 + '@smithy/util-defaults-mode-node': 3.0.25 + '@smithy/util-endpoints': 2.1.4 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-retry': 3.0.8 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + transitivePeerDependencies: + - aws-crt + + '@aws-sdk/client-sso@3.682.0': + dependencies: + '@aws-crypto/sha256-browser': 5.2.0 + '@aws-crypto/sha256-js': 5.2.0 + '@aws-sdk/core': 3.679.0 + '@aws-sdk/middleware-host-header': 3.679.0 + '@aws-sdk/middleware-logger': 3.679.0 + '@aws-sdk/middleware-recursion-detection': 3.679.0 + '@aws-sdk/middleware-user-agent': 3.682.0 + '@aws-sdk/region-config-resolver': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-endpoints': 3.679.0 + '@aws-sdk/util-user-agent-browser': 3.679.0 + '@aws-sdk/util-user-agent-node': 3.682.0 + '@smithy/config-resolver': 3.0.10 + '@smithy/core': 2.5.1 + '@smithy/fetch-http-handler': 3.2.9 + '@smithy/hash-node': 3.0.8 + '@smithy/invalid-dependency': 3.0.8 + '@smithy/middleware-content-length': 3.0.10 + '@smithy/middleware-endpoint': 3.2.1 + '@smithy/middleware-retry': 3.0.25 + '@smithy/middleware-serde': 3.0.8 + '@smithy/middleware-stack': 3.0.8 + '@smithy/node-config-provider': 3.1.9 + '@smithy/node-http-handler': 3.2.5 + '@smithy/protocol-http': 4.1.5 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/url-parser': 3.0.8 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.25 + '@smithy/util-defaults-mode-node': 3.0.25 + '@smithy/util-endpoints': 2.1.4 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-retry': 3.0.8 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + transitivePeerDependencies: + - aws-crt + + '@aws-sdk/client-sts@3.682.0': + dependencies: + '@aws-crypto/sha256-browser': 5.2.0 + '@aws-crypto/sha256-js': 5.2.0 + '@aws-sdk/client-sso-oidc': 3.682.0(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/core': 3.679.0 + '@aws-sdk/credential-provider-node': 3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/middleware-host-header': 3.679.0 + '@aws-sdk/middleware-logger': 3.679.0 + '@aws-sdk/middleware-recursion-detection': 3.679.0 + '@aws-sdk/middleware-user-agent': 3.682.0 + '@aws-sdk/region-config-resolver': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-endpoints': 3.679.0 + '@aws-sdk/util-user-agent-browser': 3.679.0 + '@aws-sdk/util-user-agent-node': 3.682.0 + '@smithy/config-resolver': 3.0.10 + '@smithy/core': 2.5.1 + '@smithy/fetch-http-handler': 3.2.9 + '@smithy/hash-node': 3.0.8 + '@smithy/invalid-dependency': 3.0.8 + '@smithy/middleware-content-length': 3.0.10 + '@smithy/middleware-endpoint': 3.2.1 + '@smithy/middleware-retry': 3.0.25 + '@smithy/middleware-serde': 3.0.8 + '@smithy/middleware-stack': 3.0.8 + '@smithy/node-config-provider': 3.1.9 + '@smithy/node-http-handler': 3.2.5 + '@smithy/protocol-http': 4.1.5 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/url-parser': 3.0.8 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.25 + '@smithy/util-defaults-mode-node': 3.0.25 + '@smithy/util-endpoints': 2.1.4 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-retry': 3.0.8 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + transitivePeerDependencies: + - aws-crt + + '@aws-sdk/core@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/core': 2.5.1 + '@smithy/node-config-provider': 3.1.9 + '@smithy/property-provider': 3.1.8 + '@smithy/protocol-http': 4.1.5 + '@smithy/signature-v4': 4.2.1 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/util-middleware': 3.0.8 + fast-xml-parser: 4.4.1 + tslib: 2.8.1 + + '@aws-sdk/credential-provider-env@3.679.0': + dependencies: + '@aws-sdk/core': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@smithy/property-provider': 3.1.8 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/credential-provider-http@3.679.0': + dependencies: + '@aws-sdk/core': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@smithy/fetch-http-handler': 3.2.9 + '@smithy/node-http-handler': 3.2.5 + '@smithy/property-provider': 3.1.8 + '@smithy/protocol-http': 4.1.5 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/util-stream': 3.2.1 + tslib: 2.8.1 + + '@aws-sdk/credential-provider-ini@3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))(@aws-sdk/client-sts@3.682.0)': + dependencies: + '@aws-sdk/client-sts': 3.682.0 + '@aws-sdk/core': 3.679.0 + '@aws-sdk/credential-provider-env': 3.679.0 + '@aws-sdk/credential-provider-http': 3.679.0 + '@aws-sdk/credential-provider-process': 3.679.0 + '@aws-sdk/credential-provider-sso': 3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0)) + '@aws-sdk/credential-provider-web-identity': 3.679.0(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/types': 3.679.0 + '@smithy/credential-provider-imds': 3.2.5 + '@smithy/property-provider': 3.1.8 + '@smithy/shared-ini-file-loader': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + transitivePeerDependencies: + - '@aws-sdk/client-sso-oidc' + - aws-crt + + '@aws-sdk/credential-provider-node@3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))(@aws-sdk/client-sts@3.682.0)': + dependencies: + '@aws-sdk/credential-provider-env': 3.679.0 + '@aws-sdk/credential-provider-http': 3.679.0 + '@aws-sdk/credential-provider-ini': 3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/credential-provider-process': 3.679.0 + '@aws-sdk/credential-provider-sso': 3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0)) + '@aws-sdk/credential-provider-web-identity': 3.679.0(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/types': 3.679.0 + '@smithy/credential-provider-imds': 3.2.5 + '@smithy/property-provider': 3.1.8 + '@smithy/shared-ini-file-loader': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + transitivePeerDependencies: + - '@aws-sdk/client-sso-oidc' + - '@aws-sdk/client-sts' + - aws-crt + + '@aws-sdk/credential-provider-process@3.679.0': + dependencies: + '@aws-sdk/core': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@smithy/property-provider': 3.1.8 + '@smithy/shared-ini-file-loader': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/credential-provider-sso@3.682.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))': + dependencies: + '@aws-sdk/client-sso': 3.682.0 + '@aws-sdk/core': 3.679.0 + '@aws-sdk/token-providers': 3.679.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0)) + '@aws-sdk/types': 3.679.0 + '@smithy/property-provider': 3.1.8 + '@smithy/shared-ini-file-loader': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + transitivePeerDependencies: + - '@aws-sdk/client-sso-oidc' + - aws-crt + + '@aws-sdk/credential-provider-web-identity@3.679.0(@aws-sdk/client-sts@3.682.0)': + dependencies: + '@aws-sdk/client-sts': 3.682.0 + '@aws-sdk/core': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@smithy/property-provider': 3.1.8 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-bucket-endpoint@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-arn-parser': 3.679.0 + '@smithy/node-config-provider': 3.1.9 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + '@smithy/util-config-provider': 3.0.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-expect-continue@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-flexible-checksums@3.682.0': + dependencies: + '@aws-crypto/crc32': 5.2.0 + '@aws-crypto/crc32c': 5.2.0 + '@aws-sdk/core': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@smithy/is-array-buffer': 3.0.0 + '@smithy/node-config-provider': 3.1.9 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-host-header@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-location-constraint@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-logger@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-recursion-detection@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-sdk-s3@3.685.0': + dependencies: + '@aws-sdk/core': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-arn-parser': 3.679.0 + '@smithy/core': 2.5.1 + '@smithy/node-config-provider': 3.1.9 + '@smithy/protocol-http': 4.1.5 + '@smithy/signature-v4': 4.2.1 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/util-config-provider': 3.0.0 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-stream': 3.2.1 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-ssec@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/middleware-user-agent@3.682.0': + dependencies: + '@aws-sdk/core': 3.679.0 + '@aws-sdk/types': 3.679.0 + '@aws-sdk/util-endpoints': 3.679.0 + '@smithy/core': 2.5.1 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/region-config-resolver@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/node-config-provider': 3.1.9 + '@smithy/types': 3.6.0 + '@smithy/util-config-provider': 3.0.0 + '@smithy/util-middleware': 3.0.8 + tslib: 2.8.1 + + '@aws-sdk/signature-v4-multi-region@3.685.0': + dependencies: + '@aws-sdk/middleware-sdk-s3': 3.685.0 + '@aws-sdk/types': 3.679.0 + '@smithy/protocol-http': 4.1.5 + '@smithy/signature-v4': 4.2.1 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/token-providers@3.679.0(@aws-sdk/client-sso-oidc@3.682.0(@aws-sdk/client-sts@3.682.0))': + dependencies: + '@aws-sdk/client-sso-oidc': 3.682.0(@aws-sdk/client-sts@3.682.0) + '@aws-sdk/types': 3.679.0 + '@smithy/property-provider': 3.1.8 + '@smithy/shared-ini-file-loader': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/types@3.679.0': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/util-arn-parser@3.679.0': + dependencies: + tslib: 2.8.1 + + '@aws-sdk/util-endpoints@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/types': 3.6.0 + '@smithy/util-endpoints': 2.1.4 + tslib: 2.8.1 + + '@aws-sdk/util-locate-window@3.679.0': + dependencies: + tslib: 2.8.1 + + '@aws-sdk/util-user-agent-browser@3.679.0': + dependencies: + '@aws-sdk/types': 3.679.0 + '@smithy/types': 3.6.0 + bowser: 2.11.0 + tslib: 2.8.1 + + '@aws-sdk/util-user-agent-node@3.682.0': + dependencies: + '@aws-sdk/middleware-user-agent': 3.682.0 + '@aws-sdk/types': 3.679.0 + '@smithy/node-config-provider': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@aws-sdk/xml-builder@3.679.0': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@babel/code-frame@7.26.2': + dependencies: + '@babel/helper-validator-identifier': 7.25.9 + js-tokens: 4.0.0 + picocolors: 1.1.0 + + '@babel/compat-data@7.26.2': {} + + '@babel/core@7.26.0': + dependencies: + '@ampproject/remapping': 2.3.0 + '@babel/code-frame': 7.26.2 + '@babel/generator': 7.26.2 + '@babel/helper-compilation-targets': 7.25.9 + '@babel/helper-module-transforms': 7.26.0(@babel/core@7.26.0) + '@babel/helpers': 7.26.0 + '@babel/parser': 7.26.2 + '@babel/template': 7.25.9 + '@babel/traverse': 7.25.9 + '@babel/types': 7.26.0 + convert-source-map: 2.0.0 + debug: 4.3.7 + gensync: 1.0.0-beta.2 + json5: 2.2.3 + semver: 6.3.1 + transitivePeerDependencies: + - supports-color + + '@babel/generator@7.26.2': + dependencies: + '@babel/parser': 7.26.2 + '@babel/types': 7.26.0 + '@jridgewell/gen-mapping': 0.3.5 + '@jridgewell/trace-mapping': 0.3.25 + jsesc: 3.0.2 + + '@babel/helper-compilation-targets@7.25.9': + dependencies: + '@babel/compat-data': 7.26.2 + '@babel/helper-validator-option': 7.25.9 + browserslist: 4.24.2 + lru-cache: 5.1.1 + semver: 6.3.1 + + '@babel/helper-module-imports@7.25.9': + dependencies: + '@babel/traverse': 7.25.9 + '@babel/types': 7.26.0 + transitivePeerDependencies: + - supports-color + + '@babel/helper-module-transforms@7.26.0(@babel/core@7.26.0)': + dependencies: + '@babel/core': 7.26.0 + '@babel/helper-module-imports': 7.25.9 + '@babel/helper-validator-identifier': 7.25.9 + '@babel/traverse': 7.25.9 + transitivePeerDependencies: + - supports-color + + '@babel/helper-string-parser@7.25.9': {} + + '@babel/helper-validator-identifier@7.25.9': {} + + '@babel/helper-validator-option@7.25.9': {} + + '@babel/helpers@7.26.0': + dependencies: + '@babel/template': 7.25.9 + '@babel/types': 7.26.0 + + '@babel/parser@7.26.2': + dependencies: + '@babel/types': 7.26.0 + + '@babel/runtime@7.26.0': + dependencies: + regenerator-runtime: 0.14.1 + + '@babel/template@7.25.9': + dependencies: + '@babel/code-frame': 7.26.2 + '@babel/parser': 7.26.2 + '@babel/types': 7.26.0 + + '@babel/traverse@7.25.9': + dependencies: + '@babel/code-frame': 7.26.2 + '@babel/generator': 7.26.2 + '@babel/parser': 7.26.2 + '@babel/template': 7.25.9 + '@babel/types': 7.26.0 + debug: 4.3.7 + globals: 11.12.0 + transitivePeerDependencies: + - supports-color + + '@babel/types@7.26.0': + dependencies: + '@babel/helper-string-parser': 7.25.9 + '@babel/helper-validator-identifier': 7.25.9 + + '@changesets/apply-release-plan@7.0.5': + dependencies: + '@changesets/config': 3.0.3 + '@changesets/get-version-range-type': 0.4.0 + '@changesets/git': 3.0.1 + '@changesets/should-skip-package': 0.1.1 + '@changesets/types': 6.0.0 + '@manypkg/get-packages': 1.1.3 + detect-indent: 6.1.0 + fs-extra: 7.0.1 + lodash.startcase: 4.4.0 + outdent: 0.5.0 + prettier: 2.8.8 + resolve-from: 5.0.0 + semver: 7.6.3 + + '@changesets/assemble-release-plan@6.0.4': + dependencies: + '@changesets/errors': 0.2.0 + '@changesets/get-dependents-graph': 2.1.2 + '@changesets/should-skip-package': 0.1.1 + '@changesets/types': 6.0.0 + '@manypkg/get-packages': 1.1.3 + semver: 7.6.3 + + '@changesets/changelog-git@0.2.0': + dependencies: + '@changesets/types': 6.0.0 + + '@changesets/cli@2.27.9': + dependencies: + '@changesets/apply-release-plan': 7.0.5 + '@changesets/assemble-release-plan': 6.0.4 + '@changesets/changelog-git': 0.2.0 + '@changesets/config': 3.0.3 + '@changesets/errors': 0.2.0 + '@changesets/get-dependents-graph': 2.1.2 + '@changesets/get-release-plan': 4.0.4 + '@changesets/git': 3.0.1 + '@changesets/logger': 0.1.1 + '@changesets/pre': 2.0.1 + '@changesets/read': 0.6.1 + '@changesets/should-skip-package': 0.1.1 + '@changesets/types': 6.0.0 + '@changesets/write': 0.3.2 + '@manypkg/get-packages': 1.1.3 + ansi-colors: 4.1.3 + ci-info: 3.9.0 + enquirer: 2.4.1 + external-editor: 3.1.0 + fs-extra: 7.0.1 + mri: 1.2.0 + p-limit: 2.3.0 + package-manager-detector: 0.2.2 + picocolors: 1.1.1 + resolve-from: 5.0.0 + semver: 7.6.3 + spawndamnit: 2.0.0 + term-size: 2.2.1 + + '@changesets/config@3.0.3': + dependencies: + '@changesets/errors': 0.2.0 + '@changesets/get-dependents-graph': 2.1.2 + '@changesets/logger': 0.1.1 + '@changesets/types': 6.0.0 + '@manypkg/get-packages': 1.1.3 + fs-extra: 7.0.1 + micromatch: 4.0.8 + + '@changesets/errors@0.2.0': + dependencies: + extendable-error: 0.1.7 + + '@changesets/get-dependents-graph@2.1.2': + dependencies: + '@changesets/types': 6.0.0 + '@manypkg/get-packages': 1.1.3 + picocolors: 1.1.1 + semver: 7.6.3 + + '@changesets/get-release-plan@4.0.4': + dependencies: + '@changesets/assemble-release-plan': 6.0.4 + '@changesets/config': 3.0.3 + '@changesets/pre': 2.0.1 + '@changesets/read': 0.6.1 + '@changesets/types': 6.0.0 + '@manypkg/get-packages': 1.1.3 + + '@changesets/get-version-range-type@0.4.0': {} + + '@changesets/git@3.0.1': + dependencies: + '@changesets/errors': 0.2.0 + '@manypkg/get-packages': 1.1.3 + is-subdir: 1.2.0 + micromatch: 4.0.8 + spawndamnit: 2.0.0 + + '@changesets/logger@0.1.1': + dependencies: + picocolors: 1.1.1 + + '@changesets/parse@0.4.0': + dependencies: + '@changesets/types': 6.0.0 + js-yaml: 3.14.1 + + '@changesets/pre@2.0.1': + dependencies: + '@changesets/errors': 0.2.0 + '@changesets/types': 6.0.0 + '@manypkg/get-packages': 1.1.3 + fs-extra: 7.0.1 + + '@changesets/read@0.6.1': + dependencies: + '@changesets/git': 3.0.1 + '@changesets/logger': 0.1.1 + '@changesets/parse': 0.4.0 + '@changesets/types': 6.0.0 + fs-extra: 7.0.1 + p-filter: 2.1.0 + picocolors: 1.1.1 + + '@changesets/should-skip-package@0.1.1': + dependencies: + '@changesets/types': 6.0.0 + '@manypkg/get-packages': 1.1.3 + + '@changesets/types@4.1.0': {} + + '@changesets/types@6.0.0': {} + + '@changesets/write@0.3.2': + dependencies: + '@changesets/types': 6.0.0 + fs-extra: 7.0.1 + human-id: 1.0.2 + prettier: 2.8.8 + + '@dependents/detective-less@5.0.0': + dependencies: + gonzales-pe: 4.3.0 + node-source-walk: 7.0.0 + + '@effect/schema@0.75.5(effect@3.10.8)': + dependencies: + effect: 3.10.8 + fast-check: 3.23.0 + optional: true + + '@esbuild/aix-ppc64@0.21.5': + optional: true + + '@esbuild/aix-ppc64@0.24.0': + optional: true + + '@esbuild/android-arm64@0.21.5': + optional: true + + '@esbuild/android-arm64@0.24.0': + optional: true + + '@esbuild/android-arm@0.21.5': + optional: true + + '@esbuild/android-arm@0.24.0': + optional: true + + '@esbuild/android-x64@0.21.5': + optional: true + + '@esbuild/android-x64@0.24.0': + optional: true + + '@esbuild/darwin-arm64@0.21.5': + optional: true + + '@esbuild/darwin-arm64@0.24.0': + optional: true + + '@esbuild/darwin-x64@0.21.5': + optional: true + + '@esbuild/darwin-x64@0.24.0': + optional: true + + '@esbuild/freebsd-arm64@0.21.5': + optional: true + + '@esbuild/freebsd-arm64@0.24.0': + optional: true + + '@esbuild/freebsd-x64@0.21.5': + optional: true + + '@esbuild/freebsd-x64@0.24.0': + optional: true + + '@esbuild/linux-arm64@0.21.5': + optional: true + + '@esbuild/linux-arm64@0.24.0': + optional: true + + '@esbuild/linux-arm@0.21.5': + optional: true + + '@esbuild/linux-arm@0.24.0': + optional: true + + '@esbuild/linux-ia32@0.21.5': + optional: true + + '@esbuild/linux-ia32@0.24.0': + optional: true + + '@esbuild/linux-loong64@0.21.5': + optional: true + + '@esbuild/linux-loong64@0.24.0': + optional: true + + '@esbuild/linux-mips64el@0.21.5': + optional: true + + '@esbuild/linux-mips64el@0.24.0': + optional: true + + '@esbuild/linux-ppc64@0.21.5': + optional: true + + '@esbuild/linux-ppc64@0.24.0': + optional: true + + '@esbuild/linux-riscv64@0.21.5': + optional: true + + '@esbuild/linux-riscv64@0.24.0': + optional: true + + '@esbuild/linux-s390x@0.21.5': + optional: true + + '@esbuild/linux-s390x@0.24.0': + optional: true + + '@esbuild/linux-x64@0.21.5': + optional: true + + '@esbuild/linux-x64@0.24.0': + optional: true + + '@esbuild/netbsd-x64@0.21.5': + optional: true + + '@esbuild/netbsd-x64@0.24.0': + optional: true + + '@esbuild/openbsd-arm64@0.24.0': + optional: true + + '@esbuild/openbsd-x64@0.21.5': + optional: true + + '@esbuild/openbsd-x64@0.24.0': + optional: true + + '@esbuild/sunos-x64@0.21.5': + optional: true + + '@esbuild/sunos-x64@0.24.0': + optional: true + + '@esbuild/win32-arm64@0.21.5': + optional: true + + '@esbuild/win32-arm64@0.24.0': + optional: true + + '@esbuild/win32-ia32@0.21.5': + optional: true + + '@esbuild/win32-ia32@0.24.0': + optional: true + + '@esbuild/win32-x64@0.21.5': + optional: true + + '@esbuild/win32-x64@0.24.0': + optional: true + + '@eslint-community/eslint-utils@4.4.1(eslint@8.57.1)': + dependencies: + eslint: 8.57.1 + eslint-visitor-keys: 3.4.3 + + '@eslint-community/regexpp@4.12.1': {} + + '@eslint/eslintrc@2.1.4': + dependencies: + ajv: 6.12.6 + debug: 4.3.7 + espree: 9.6.1 + globals: 13.24.0 + ignore: 5.3.2 + import-fresh: 3.3.0 + js-yaml: 4.1.0 + minimatch: 3.1.2 + strip-json-comments: 3.1.1 + transitivePeerDependencies: + - supports-color + + '@eslint/js@8.57.1': {} + + '@exodus/schemasafe@1.3.0': + optional: true + + '@floating-ui/core@1.6.8': + dependencies: + '@floating-ui/utils': 0.2.8 + + '@floating-ui/dom@1.6.12': + dependencies: + '@floating-ui/core': 1.6.8 + '@floating-ui/utils': 0.2.8 + + '@floating-ui/utils@0.2.8': {} + + '@gcornut/valibot-json-schema@0.31.0': + dependencies: + valibot: 0.31.1 + optionalDependencies: + '@types/json-schema': 7.0.15 + esbuild: 0.24.0 + esbuild-runner: 2.2.2(esbuild@0.24.0) + optional: true + + '@gcornut/valibot-json-schema@0.42.0(esbuild@0.24.0)(typescript@5.5.4)': + dependencies: + valibot: 0.42.1(typescript@5.5.4) + optionalDependencies: + '@types/json-schema': 7.0.15 + esbuild-runner: 2.2.2(esbuild@0.24.0) + transitivePeerDependencies: + - esbuild + - typescript + + '@hapi/hoek@9.3.0': + optional: true + + '@hapi/topo@5.1.0': + dependencies: + '@hapi/hoek': 9.3.0 + optional: true + + '@huakunshen/comlink@4.4.1': {} + + '@humanwhocodes/config-array@0.13.0': + dependencies: + '@humanwhocodes/object-schema': 2.0.3 + debug: 4.3.7 + minimatch: 3.1.2 + transitivePeerDependencies: + - supports-color + + '@humanwhocodes/module-importer@1.0.1': {} + + '@humanwhocodes/object-schema@2.0.3': {} + + '@ianvs/prettier-plugin-sort-imports@4.3.1(@vue/compiler-sfc@3.5.12)(prettier@3.3.3)': + dependencies: + '@babel/core': 7.26.0 + '@babel/generator': 7.26.2 + '@babel/parser': 7.26.2 + '@babel/traverse': 7.25.9 + '@babel/types': 7.26.0 + prettier: 3.3.3 + semver: 7.6.3 + optionalDependencies: + '@vue/compiler-sfc': 3.5.12 + transitivePeerDependencies: + - supports-color + + '@iconify/svelte@4.0.2(svelte@5.1.9)': + dependencies: + '@iconify/types': 2.0.0 + svelte: 5.1.9 + + '@iconify/types@2.0.0': {} + + '@internationalized/date@3.5.6': + dependencies: + '@swc/helpers': 0.5.13 + + '@isaacs/cliui@8.0.2': + dependencies: + string-width: 5.1.2 + string-width-cjs: string-width@4.2.3 + strip-ansi: 7.1.0 + strip-ansi-cjs: strip-ansi@6.0.1 + wrap-ansi: 8.1.0 + wrap-ansi-cjs: wrap-ansi@7.0.0 + + '@isaacs/fs-minipass@4.0.1': + dependencies: + minipass: 7.1.2 + + '@jridgewell/gen-mapping@0.3.5': + dependencies: + '@jridgewell/set-array': 1.2.1 + '@jridgewell/sourcemap-codec': 1.5.0 + '@jridgewell/trace-mapping': 0.3.25 + + '@jridgewell/resolve-uri@3.1.2': {} + + '@jridgewell/set-array@1.2.1': {} + + '@jridgewell/source-map@0.3.6': + dependencies: + '@jridgewell/gen-mapping': 0.3.5 + '@jridgewell/trace-mapping': 0.3.25 + optional: true + + '@jridgewell/sourcemap-codec@1.5.0': {} + + '@jridgewell/trace-mapping@0.3.25': + dependencies: + '@jridgewell/resolve-uri': 3.1.2 + '@jridgewell/sourcemap-codec': 1.5.0 + + '@jsr/std__semver@1.0.3': {} + + '@kksh/svelte5@0.1.2-beta.4(lucide-svelte@0.454.0(svelte@5.1.9))(svelte-sonner@0.3.28(svelte@5.1.9))(svelte@5.1.9)': + dependencies: + lucide-svelte: 0.454.0(svelte@5.1.9) + svelte: 5.1.9 + svelte-persisted-store: 0.11.0(svelte@5.1.9) + svelte-sonner: 0.3.28(svelte@5.1.9) + + '@manypkg/find-root@1.1.0': + dependencies: + '@babel/runtime': 7.26.0 + '@types/node': 12.20.55 + find-up: 4.1.0 + fs-extra: 8.1.0 + + '@manypkg/get-packages@1.1.3': + dependencies: + '@babel/runtime': 7.26.0 + '@changesets/types': 4.1.0 + '@manypkg/find-root': 1.1.0 + fs-extra: 8.1.0 + globby: 11.1.0 + read-yaml-file: 1.1.0 + + '@melt-ui/svelte@0.76.2(svelte@5.1.9)': + dependencies: + '@floating-ui/core': 1.6.8 + '@floating-ui/dom': 1.6.12 + '@internationalized/date': 3.5.6 + dequal: 2.0.3 + focus-trap: 7.6.0 + nanoid: 5.0.8 + svelte: 5.1.9 + + '@nodelib/fs.scandir@2.1.5': + dependencies: + '@nodelib/fs.stat': 2.0.5 + run-parallel: 1.2.0 + + '@nodelib/fs.stat@2.0.5': {} + + '@nodelib/fs.walk@1.2.8': + dependencies: + '@nodelib/fs.scandir': 2.1.5 + fastq: 1.17.1 + + '@pkgjs/parseargs@0.11.0': + optional: true + + '@polka/url@1.0.0-next.28': {} + + '@poppinss/macroable@1.0.3': + optional: true + + '@rollup/plugin-alias@5.1.1(rollup@4.24.3)': + optionalDependencies: + rollup: 4.24.3 + + '@rollup/plugin-typescript@11.1.6(rollup@4.24.3)(tslib@2.8.1)(typescript@5.5.4)': + dependencies: + '@rollup/pluginutils': 5.1.3(rollup@4.24.3) + resolve: 1.22.8 + typescript: 5.5.4 + optionalDependencies: + rollup: 4.24.3 + tslib: 2.8.1 + + '@rollup/plugin-typescript@11.1.6(rollup@4.24.3)(tslib@2.8.1)(typescript@5.6.3)': + dependencies: + '@rollup/pluginutils': 5.1.3(rollup@4.24.3) + resolve: 1.22.8 + typescript: 5.6.3 + optionalDependencies: + rollup: 4.24.3 + tslib: 2.8.1 + + '@rollup/pluginutils@5.1.3(rollup@4.24.3)': + dependencies: + '@types/estree': 1.0.6 + estree-walker: 2.0.2 + picomatch: 4.0.2 + optionalDependencies: + rollup: 4.24.3 + + '@rollup/rollup-android-arm-eabi@4.24.3': + optional: true + + '@rollup/rollup-android-arm64@4.24.3': + optional: true + + '@rollup/rollup-darwin-arm64@4.24.3': + optional: true + + '@rollup/rollup-darwin-x64@4.24.3': + optional: true + + '@rollup/rollup-freebsd-arm64@4.24.3': + optional: true + + '@rollup/rollup-freebsd-x64@4.24.3': + optional: true + + '@rollup/rollup-linux-arm-gnueabihf@4.24.3': + optional: true + + '@rollup/rollup-linux-arm-musleabihf@4.24.3': + optional: true + + '@rollup/rollup-linux-arm64-gnu@4.24.3': + optional: true + + '@rollup/rollup-linux-arm64-musl@4.24.3': + optional: true + + '@rollup/rollup-linux-powerpc64le-gnu@4.24.3': + optional: true + + '@rollup/rollup-linux-riscv64-gnu@4.24.3': + optional: true + + '@rollup/rollup-linux-s390x-gnu@4.24.3': + optional: true + + '@rollup/rollup-linux-x64-gnu@4.24.3': + optional: true + + '@rollup/rollup-linux-x64-musl@4.24.3': + optional: true + + '@rollup/rollup-win32-arm64-msvc@4.24.3': + optional: true + + '@rollup/rollup-win32-ia32-msvc@4.24.3': + optional: true + + '@rollup/rollup-win32-x64-msvc@4.24.3': + optional: true + + '@shikijs/core@1.22.2': + dependencies: + '@shikijs/engine-javascript': 1.22.2 + '@shikijs/engine-oniguruma': 1.22.2 + '@shikijs/types': 1.22.2 + '@shikijs/vscode-textmate': 9.3.0 + '@types/hast': 3.0.4 + hast-util-to-html: 9.0.3 + + '@shikijs/engine-javascript@1.22.2': + dependencies: + '@shikijs/types': 1.22.2 + '@shikijs/vscode-textmate': 9.3.0 + oniguruma-to-js: 0.4.3 + + '@shikijs/engine-oniguruma@1.22.2': + dependencies: + '@shikijs/types': 1.22.2 + '@shikijs/vscode-textmate': 9.3.0 + + '@shikijs/types@1.22.2': + dependencies: + '@shikijs/vscode-textmate': 9.3.0 + '@types/hast': 3.0.4 + + '@shikijs/vscode-textmate@9.3.0': {} + + '@sideway/address@4.1.5': + dependencies: + '@hapi/hoek': 9.3.0 + optional: true + + '@sideway/formula@3.0.1': + optional: true + + '@sideway/pinpoint@2.0.0': + optional: true + + '@sinclair/typebox@0.32.35': + optional: true + + '@smithy/abort-controller@3.1.6': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/chunked-blob-reader-native@3.0.1': + dependencies: + '@smithy/util-base64': 3.0.0 + tslib: 2.8.1 + + '@smithy/chunked-blob-reader@4.0.0': + dependencies: + tslib: 2.8.1 + + '@smithy/config-resolver@3.0.10': + dependencies: + '@smithy/node-config-provider': 3.1.9 + '@smithy/types': 3.6.0 + '@smithy/util-config-provider': 3.0.0 + '@smithy/util-middleware': 3.0.8 + tslib: 2.8.1 + + '@smithy/core@2.5.1': + dependencies: + '@smithy/middleware-serde': 3.0.8 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-stream': 3.2.1 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@smithy/credential-provider-imds@3.2.5': + dependencies: + '@smithy/node-config-provider': 3.1.9 + '@smithy/property-provider': 3.1.8 + '@smithy/types': 3.6.0 + '@smithy/url-parser': 3.0.8 + tslib: 2.8.1 + + '@smithy/eventstream-codec@3.1.7': + dependencies: + '@aws-crypto/crc32': 5.2.0 + '@smithy/types': 3.6.0 + '@smithy/util-hex-encoding': 3.0.0 + tslib: 2.8.1 + + '@smithy/eventstream-serde-browser@3.0.11': + dependencies: + '@smithy/eventstream-serde-universal': 3.0.10 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/eventstream-serde-config-resolver@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/eventstream-serde-node@3.0.10': + dependencies: + '@smithy/eventstream-serde-universal': 3.0.10 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/eventstream-serde-universal@3.0.10': + dependencies: + '@smithy/eventstream-codec': 3.1.7 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/fetch-http-handler@3.2.9': + dependencies: + '@smithy/protocol-http': 4.1.5 + '@smithy/querystring-builder': 3.0.8 + '@smithy/types': 3.6.0 + '@smithy/util-base64': 3.0.0 + tslib: 2.8.1 + + '@smithy/fetch-http-handler@4.0.0': + dependencies: + '@smithy/protocol-http': 4.1.5 + '@smithy/querystring-builder': 3.0.8 + '@smithy/types': 3.6.0 + '@smithy/util-base64': 3.0.0 + tslib: 2.8.1 + + '@smithy/hash-blob-browser@3.1.7': + dependencies: + '@smithy/chunked-blob-reader': 4.0.0 + '@smithy/chunked-blob-reader-native': 3.0.1 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/hash-node@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + '@smithy/util-buffer-from': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@smithy/hash-stream-node@3.1.7': + dependencies: + '@smithy/types': 3.6.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@smithy/invalid-dependency@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/is-array-buffer@2.2.0': + dependencies: + tslib: 2.8.1 + + '@smithy/is-array-buffer@3.0.0': + dependencies: + tslib: 2.8.1 + + '@smithy/md5-js@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@smithy/middleware-content-length@3.0.10': + dependencies: + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/middleware-endpoint@3.2.1': + dependencies: + '@smithy/core': 2.5.1 + '@smithy/middleware-serde': 3.0.8 + '@smithy/node-config-provider': 3.1.9 + '@smithy/shared-ini-file-loader': 3.1.9 + '@smithy/types': 3.6.0 + '@smithy/url-parser': 3.0.8 + '@smithy/util-middleware': 3.0.8 + tslib: 2.8.1 + + '@smithy/middleware-retry@3.0.25': + dependencies: + '@smithy/node-config-provider': 3.1.9 + '@smithy/protocol-http': 4.1.5 + '@smithy/service-error-classification': 3.0.8 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-retry': 3.0.8 + tslib: 2.8.1 + uuid: 9.0.1 + + '@smithy/middleware-serde@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/middleware-stack@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/node-config-provider@3.1.9': + dependencies: + '@smithy/property-provider': 3.1.8 + '@smithy/shared-ini-file-loader': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/node-http-handler@3.2.5': + dependencies: + '@smithy/abort-controller': 3.1.6 + '@smithy/protocol-http': 4.1.5 + '@smithy/querystring-builder': 3.0.8 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/property-provider@3.1.8': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/protocol-http@4.1.5': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/querystring-builder@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + '@smithy/util-uri-escape': 3.0.0 + tslib: 2.8.1 + + '@smithy/querystring-parser@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/service-error-classification@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + + '@smithy/shared-ini-file-loader@3.1.9': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/signature-v4@4.2.1': + dependencies: + '@smithy/is-array-buffer': 3.0.0 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + '@smithy/util-hex-encoding': 3.0.0 + '@smithy/util-middleware': 3.0.8 + '@smithy/util-uri-escape': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@smithy/smithy-client@3.4.2': + dependencies: + '@smithy/core': 2.5.1 + '@smithy/middleware-endpoint': 3.2.1 + '@smithy/middleware-stack': 3.0.8 + '@smithy/protocol-http': 4.1.5 + '@smithy/types': 3.6.0 + '@smithy/util-stream': 3.2.1 + tslib: 2.8.1 + + '@smithy/types@3.6.0': + dependencies: + tslib: 2.8.1 + + '@smithy/url-parser@3.0.8': + dependencies: + '@smithy/querystring-parser': 3.0.8 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/util-base64@3.0.0': + dependencies: + '@smithy/util-buffer-from': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@smithy/util-body-length-browser@3.0.0': + dependencies: + tslib: 2.8.1 + + '@smithy/util-body-length-node@3.0.0': + dependencies: + tslib: 2.8.1 + + '@smithy/util-buffer-from@2.2.0': + dependencies: + '@smithy/is-array-buffer': 2.2.0 + tslib: 2.8.1 + + '@smithy/util-buffer-from@3.0.0': + dependencies: + '@smithy/is-array-buffer': 3.0.0 + tslib: 2.8.1 + + '@smithy/util-config-provider@3.0.0': + dependencies: + tslib: 2.8.1 + + '@smithy/util-defaults-mode-browser@3.0.25': + dependencies: + '@smithy/property-provider': 3.1.8 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + bowser: 2.11.0 + tslib: 2.8.1 + + '@smithy/util-defaults-mode-node@3.0.25': + dependencies: + '@smithy/config-resolver': 3.0.10 + '@smithy/credential-provider-imds': 3.2.5 + '@smithy/node-config-provider': 3.1.9 + '@smithy/property-provider': 3.1.8 + '@smithy/smithy-client': 3.4.2 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/util-endpoints@2.1.4': + dependencies: + '@smithy/node-config-provider': 3.1.9 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/util-hex-encoding@3.0.0': + dependencies: + tslib: 2.8.1 + + '@smithy/util-middleware@3.0.8': + dependencies: + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/util-retry@3.0.8': + dependencies: + '@smithy/service-error-classification': 3.0.8 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@smithy/util-stream@3.2.1': + dependencies: + '@smithy/fetch-http-handler': 4.0.0 + '@smithy/node-http-handler': 3.2.5 + '@smithy/types': 3.6.0 + '@smithy/util-base64': 3.0.0 + '@smithy/util-buffer-from': 3.0.0 + '@smithy/util-hex-encoding': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.8.1 + + '@smithy/util-uri-escape@3.0.0': + dependencies: + tslib: 2.8.1 + + '@smithy/util-utf8@2.3.0': + dependencies: + '@smithy/util-buffer-from': 2.2.0 + tslib: 2.8.1 + + '@smithy/util-utf8@3.0.0': + dependencies: + '@smithy/util-buffer-from': 3.0.0 + tslib: 2.8.1 + + '@smithy/util-waiter@3.1.7': + dependencies: + '@smithy/abort-controller': 3.1.6 + '@smithy/types': 3.6.0 + tslib: 2.8.1 + + '@supabase/auth-js@2.65.1': + dependencies: + '@supabase/node-fetch': 2.6.15 + + '@supabase/functions-js@2.4.3': + dependencies: + '@supabase/node-fetch': 2.6.15 + + '@supabase/node-fetch@2.6.15': + dependencies: + whatwg-url: 5.0.0 + + '@supabase/postgrest-js@1.16.3': + dependencies: + '@supabase/node-fetch': 2.6.15 + + '@supabase/realtime-js@2.10.7': + dependencies: + '@supabase/node-fetch': 2.6.15 + '@types/phoenix': 1.6.5 + '@types/ws': 8.5.13 + ws: 8.18.0 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + + '@supabase/storage-js@2.7.1': + dependencies: + '@supabase/node-fetch': 2.6.15 + + '@supabase/supabase-js@2.46.1': + dependencies: + '@supabase/auth-js': 2.65.1 + '@supabase/functions-js': 2.4.3 + '@supabase/node-fetch': 2.6.15 + '@supabase/postgrest-js': 1.16.3 + '@supabase/realtime-js': 2.10.7 + '@supabase/storage-js': 2.7.1 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + + '@sveltejs/adapter-static@3.0.6(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))': + dependencies: + '@sveltejs/kit': 2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) + + '@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0))': + dependencies: + '@sveltejs/vite-plugin-svelte': 4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) + '@types/cookie': 0.6.0 + cookie: 0.6.0 + devalue: 5.1.1 + esm-env: 1.1.4 + import-meta-resolve: 4.1.0 + kleur: 4.1.5 + magic-string: 0.30.12 + mrmime: 2.0.0 + sade: 1.8.1 + set-cookie-parser: 2.7.1 + sirv: 3.0.0 + svelte: 5.1.9 + tiny-glob: 0.2.9 + vite: 5.4.10(@types/node@22.8.7)(terser@5.36.0) + + '@sveltejs/vite-plugin-svelte-inspector@3.0.1(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0))': + dependencies: + '@sveltejs/vite-plugin-svelte': 4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) + debug: 4.3.7 + svelte: 5.1.9 + vite: 5.4.10(@types/node@22.8.7)(terser@5.36.0) + transitivePeerDependencies: + - supports-color + + '@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0))': + dependencies: + '@sveltejs/vite-plugin-svelte-inspector': 3.0.1(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) + debug: 4.3.7 + deepmerge: 4.3.1 + kleur: 4.1.5 + magic-string: 0.30.12 + svelte: 5.1.9 + vite: 5.4.10(@types/node@22.8.7)(terser@5.36.0) + vitefu: 1.0.3(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) + transitivePeerDependencies: + - supports-color + + '@swc/helpers@0.5.13': + dependencies: + tslib: 2.8.1 + + '@tailwindcss/aspect-ratio@0.4.2(tailwindcss@3.4.14)': + dependencies: + tailwindcss: 3.4.14 + + '@tailwindcss/container-queries@0.1.1(tailwindcss@3.4.14)': + dependencies: + tailwindcss: 3.4.14 + + '@tailwindcss/forms@0.5.9(tailwindcss@3.4.14)': + dependencies: + mini-svg-data-uri: 1.4.4 + tailwindcss: 3.4.14 + + '@tailwindcss/typography@0.5.15(tailwindcss@3.4.14)': + dependencies: + lodash.castarray: 4.4.0 + lodash.isplainobject: 4.0.6 + lodash.merge: 4.6.2 + postcss-selector-parser: 6.0.10 + tailwindcss: 3.4.14 + + '@tauri-apps/api@2.0.1': {} + + '@tauri-apps/api@2.0.3': {} + + '@tauri-apps/cli-darwin-arm64@2.0.4': + optional: true + + '@tauri-apps/cli-darwin-x64@2.0.4': + optional: true + + '@tauri-apps/cli-linux-arm-gnueabihf@2.0.4': + optional: true + + '@tauri-apps/cli-linux-arm64-gnu@2.0.4': + optional: true + + '@tauri-apps/cli-linux-arm64-musl@2.0.4': + optional: true + + '@tauri-apps/cli-linux-x64-gnu@2.0.4': + optional: true + + '@tauri-apps/cli-linux-x64-musl@2.0.4': + optional: true + + '@tauri-apps/cli-win32-arm64-msvc@2.0.4': + optional: true + + '@tauri-apps/cli-win32-ia32-msvc@2.0.4': + optional: true + + '@tauri-apps/cli-win32-x64-msvc@2.0.4': + optional: true + + '@tauri-apps/cli@2.0.4': + optionalDependencies: + '@tauri-apps/cli-darwin-arm64': 2.0.4 + '@tauri-apps/cli-darwin-x64': 2.0.4 + '@tauri-apps/cli-linux-arm-gnueabihf': 2.0.4 + '@tauri-apps/cli-linux-arm64-gnu': 2.0.4 + '@tauri-apps/cli-linux-arm64-musl': 2.0.4 + '@tauri-apps/cli-linux-x64-gnu': 2.0.4 + '@tauri-apps/cli-linux-x64-musl': 2.0.4 + '@tauri-apps/cli-win32-arm64-msvc': 2.0.4 + '@tauri-apps/cli-win32-ia32-msvc': 2.0.4 + '@tauri-apps/cli-win32-x64-msvc': 2.0.4 + + '@tauri-apps/plugin-deep-link@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-dialog@2.0.1': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-fs@2.0.1': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-global-shortcut@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-http@2.0.1': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-log@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-notification@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-os@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-process@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-shell@2.0.1': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-store@2.1.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-updater@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@tauri-apps/plugin-upload@2.0.0': + dependencies: + '@tauri-apps/api': 2.0.3 + + '@ts-graphviz/adapter@2.0.5': + dependencies: + '@ts-graphviz/common': 2.1.4 + + '@ts-graphviz/ast@2.0.5': + dependencies: + '@ts-graphviz/common': 2.1.4 + + '@ts-graphviz/common@2.1.4': {} + + '@ts-graphviz/core@2.0.5': + dependencies: + '@ts-graphviz/ast': 2.0.5 + '@ts-graphviz/common': 2.1.4 + + '@types/bun@1.1.13': + dependencies: + bun-types: 1.1.34 + + '@types/cookie@0.6.0': {} + + '@types/estree@1.0.6': {} + + '@types/hast@3.0.4': + dependencies: + '@types/unist': 3.0.3 + + '@types/json-schema@7.0.15': + optional: true + + '@types/lodash@4.17.13': {} + + '@types/madge@5.0.3': + dependencies: + '@types/node': 22.8.7 + + '@types/mdast@4.0.4': + dependencies: + '@types/unist': 3.0.3 + + '@types/node@12.20.55': {} + + '@types/node@20.12.14': + dependencies: + undici-types: 5.26.5 + + '@types/node@22.8.7': + dependencies: + undici-types: 6.19.8 + + '@types/phoenix@1.6.5': {} + + '@types/semver@7.5.8': {} + + '@types/unist@3.0.3': {} + + '@types/validator@13.12.2': + optional: true + + '@types/ws@8.5.13': + dependencies: + '@types/node': 22.8.7 + + '@typeschema/class-validator@0.3.0(@types/json-schema@7.0.15)(class-validator@0.14.1)': + dependencies: + '@typeschema/core': 0.14.0(@types/json-schema@7.0.15) + optionalDependencies: + class-validator: 0.14.1 + transitivePeerDependencies: + - '@types/json-schema' + optional: true + + '@typeschema/core@0.14.0(@types/json-schema@7.0.15)': + optionalDependencies: + '@types/json-schema': 7.0.15 + optional: true + + '@typescript-eslint/eslint-plugin@7.18.0(@typescript-eslint/parser@7.18.0(eslint@8.57.1)(typescript@5.6.3))(eslint@8.57.1)(typescript@5.6.3)': + dependencies: + '@eslint-community/regexpp': 4.12.1 + '@typescript-eslint/parser': 7.18.0(eslint@8.57.1)(typescript@5.6.3) + '@typescript-eslint/scope-manager': 7.18.0 + '@typescript-eslint/type-utils': 7.18.0(eslint@8.57.1)(typescript@5.6.3) + '@typescript-eslint/utils': 7.18.0(eslint@8.57.1)(typescript@5.6.3) + '@typescript-eslint/visitor-keys': 7.18.0 + eslint: 8.57.1 + graphemer: 1.4.0 + ignore: 5.3.2 + natural-compare: 1.4.0 + ts-api-utils: 1.4.0(typescript@5.6.3) + optionalDependencies: + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/parser@7.18.0(eslint@8.57.1)(typescript@5.6.3)': + dependencies: + '@typescript-eslint/scope-manager': 7.18.0 + '@typescript-eslint/types': 7.18.0 + '@typescript-eslint/typescript-estree': 7.18.0(typescript@5.6.3) + '@typescript-eslint/visitor-keys': 7.18.0 + debug: 4.3.7 + eslint: 8.57.1 + optionalDependencies: + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/scope-manager@7.18.0': + dependencies: + '@typescript-eslint/types': 7.18.0 + '@typescript-eslint/visitor-keys': 7.18.0 + + '@typescript-eslint/type-utils@7.18.0(eslint@8.57.1)(typescript@5.6.3)': + dependencies: + '@typescript-eslint/typescript-estree': 7.18.0(typescript@5.6.3) + '@typescript-eslint/utils': 7.18.0(eslint@8.57.1)(typescript@5.6.3) + debug: 4.3.7 + eslint: 8.57.1 + ts-api-utils: 1.4.0(typescript@5.6.3) + optionalDependencies: + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/types@7.18.0': {} + + '@typescript-eslint/typescript-estree@7.18.0(typescript@5.6.3)': + dependencies: + '@typescript-eslint/types': 7.18.0 + '@typescript-eslint/visitor-keys': 7.18.0 + debug: 4.3.7 + globby: 11.1.0 + is-glob: 4.0.3 + minimatch: 9.0.5 + semver: 7.6.3 + ts-api-utils: 1.4.0(typescript@5.6.3) + optionalDependencies: + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + '@typescript-eslint/utils@7.18.0(eslint@8.57.1)(typescript@5.6.3)': + dependencies: + '@eslint-community/eslint-utils': 4.4.1(eslint@8.57.1) + '@typescript-eslint/scope-manager': 7.18.0 + '@typescript-eslint/types': 7.18.0 + '@typescript-eslint/typescript-estree': 7.18.0(typescript@5.6.3) + eslint: 8.57.1 + transitivePeerDependencies: + - supports-color + - typescript + + '@typescript-eslint/visitor-keys@7.18.0': + dependencies: + '@typescript-eslint/types': 7.18.0 + eslint-visitor-keys: 3.4.3 + + '@ungap/structured-clone@1.2.0': {} + + '@vinejs/compiler@2.5.0': + optional: true + + '@vinejs/vine@1.8.0': + dependencies: + '@poppinss/macroable': 1.0.3 + '@types/validator': 13.12.2 + '@vinejs/compiler': 2.5.0 + camelcase: 8.0.0 + dayjs: 1.11.13 + dlv: 1.1.3 + normalize-url: 8.0.1 + validator: 13.12.0 + optional: true + + '@vue/compiler-core@3.5.12': + dependencies: + '@babel/parser': 7.26.2 + '@vue/shared': 3.5.12 + entities: 4.5.0 + estree-walker: 2.0.2 + source-map-js: 1.2.1 + + '@vue/compiler-dom@3.5.12': + dependencies: + '@vue/compiler-core': 3.5.12 + '@vue/shared': 3.5.12 + + '@vue/compiler-sfc@3.5.12': + dependencies: + '@babel/parser': 7.26.2 + '@vue/compiler-core': 3.5.12 + '@vue/compiler-dom': 3.5.12 + '@vue/compiler-ssr': 3.5.12 + '@vue/shared': 3.5.12 + estree-walker: 2.0.2 + magic-string: 0.30.12 + postcss: 8.4.47 + source-map-js: 1.2.1 + + '@vue/compiler-ssr@3.5.12': + dependencies: + '@vue/compiler-dom': 3.5.12 + '@vue/shared': 3.5.12 + + '@vue/shared@3.5.12': {} + + acorn-jsx@5.3.2(acorn@8.12.1): + dependencies: + acorn: 8.12.1 + + acorn-typescript@1.4.13(acorn@8.12.1): + dependencies: + acorn: 8.12.1 + + acorn@8.12.1: {} + + agent-base@7.1.1: + dependencies: + debug: 4.3.7 + transitivePeerDependencies: + - supports-color + + ajv@6.12.6: + dependencies: + fast-deep-equal: 3.1.3 + fast-json-stable-stringify: 2.1.0 + json-schema-traverse: 0.4.1 + uri-js: 4.4.1 + + ansi-colors@4.1.3: {} + + ansi-regex@5.0.1: {} + + ansi-regex@6.1.0: {} + + ansi-styles@4.3.0: + dependencies: + color-convert: 2.0.1 + + ansi-styles@6.2.1: {} + + any-promise@1.3.0: {} + + anymatch@3.1.3: + dependencies: + normalize-path: 3.0.0 + picomatch: 2.3.1 + + app-module-path@2.2.0: {} + + arg@5.0.2: {} + + argparse@1.0.10: + dependencies: + sprintf-js: 1.0.3 + + argparse@2.0.1: {} + + aria-query@5.3.2: {} + + arktype@2.0.0-rc.8: + dependencies: + '@ark/schema': 0.10.0 + '@ark/util': 0.10.0 + optional: true + + array-union@2.1.0: {} + + ast-module-types@6.0.0: {} + + autoprefixer@10.4.20(postcss@8.4.47): + dependencies: + browserslist: 4.24.2 + caniuse-lite: 1.0.30001676 + fraction.js: 4.3.7 + normalize-range: 0.1.2 + picocolors: 1.1.0 + postcss: 8.4.47 + postcss-value-parser: 4.2.0 + + axobject-query@4.1.0: {} + + balanced-match@1.0.2: {} + + base64-js@1.5.1: {} + + better-path-resolve@1.0.0: + dependencies: + is-windows: 1.0.2 + + bin-links@5.0.0: + dependencies: + cmd-shim: 7.0.0 + npm-normalize-package-bin: 4.0.0 + proc-log: 5.0.0 + read-cmd-shim: 5.0.0 + write-file-atomic: 6.0.0 + + binary-extensions@2.3.0: {} + + bits-ui@0.21.16(svelte@5.1.9): + dependencies: + '@internationalized/date': 3.5.6 + '@melt-ui/svelte': 0.76.2(svelte@5.1.9) + nanoid: 5.0.8 + svelte: 5.1.9 + + bits-ui@1.0.0-next.36(svelte@5.1.9): + dependencies: + '@floating-ui/core': 1.6.8 + '@floating-ui/dom': 1.6.12 + '@internationalized/date': 3.5.6 + esm-env: 1.1.4 + runed: 0.15.3(svelte@5.1.9) + svelte: 5.1.9 + svelte-toolbelt: 0.4.6(svelte@5.1.9) + + bl@4.1.0: + dependencies: + buffer: 5.7.1 + inherits: 2.0.4 + readable-stream: 3.6.2 + + bowser@2.11.0: {} + + brace-expansion@1.1.11: + dependencies: + balanced-match: 1.0.2 + concat-map: 0.0.1 + + brace-expansion@2.0.1: + dependencies: + balanced-match: 1.0.2 + + braces@3.0.3: + dependencies: + fill-range: 7.1.1 + + browserslist@4.24.2: + dependencies: + caniuse-lite: 1.0.30001676 + electron-to-chromium: 1.5.50 + node-releases: 2.0.18 + update-browserslist-db: 1.1.1(browserslist@4.24.2) + + buffer-from@1.1.2: + optional: true + + buffer@5.7.1: + dependencies: + base64-js: 1.5.1 + ieee754: 1.2.1 + + bun-types@1.1.34: + dependencies: + '@types/node': 20.12.14 + '@types/ws': 8.5.13 + + callsites@3.1.0: {} + + camelcase-css@2.0.1: {} + + camelcase@8.0.0: + optional: true + + caniuse-lite@1.0.30001676: {} + + ccount@2.0.1: {} + + chalk@4.1.2: + dependencies: + ansi-styles: 4.3.0 + supports-color: 7.2.0 + + character-entities-html4@2.1.0: {} + + character-entities-legacy@3.0.0: {} + + chardet@0.7.0: {} + + chokidar@3.6.0: + dependencies: + anymatch: 3.1.3 + braces: 3.0.3 + glob-parent: 5.1.2 + is-binary-path: 2.1.0 + is-glob: 4.0.3 + normalize-path: 3.0.0 + readdirp: 3.6.0 + optionalDependencies: + fsevents: 2.3.3 + + chokidar@4.0.1: + dependencies: + readdirp: 4.0.2 + + chownr@3.0.0: {} + + ci-info@3.9.0: {} + + class-validator@0.14.1: + dependencies: + '@types/validator': 13.12.2 + libphonenumber-js: 1.11.12 + validator: 13.12.0 + optional: true + + cli-cursor@3.1.0: + dependencies: + restore-cursor: 3.1.0 + + cli-spinners@2.9.2: {} + + clone@1.0.4: {} + + clsx@2.1.1: {} + + cmd-shim@7.0.0: {} + + color-convert@2.0.1: + dependencies: + color-name: 1.1.4 + + color-name@1.1.4: {} + + comlink-stdio@0.1.7(typescript@5.5.4): + dependencies: + typescript: 5.5.4 + + comlink-stdio@0.1.7(typescript@5.6.3): + dependencies: + typescript: 5.6.3 + + comlink@4.4.1: {} + + comma-separated-tokens@2.0.3: {} + + commander@12.1.0: {} + + commander@2.20.3: + optional: true + + commander@4.1.1: {} + + commander@7.2.0: {} + + commander@9.5.0: {} + + commondir@1.0.1: {} + + concat-map@0.0.1: {} + + convert-source-map@2.0.0: {} + + cookie@0.6.0: {} + + cross-spawn@5.1.0: + dependencies: + lru-cache: 4.1.5 + shebang-command: 1.2.0 + which: 1.3.1 + + cross-spawn@7.0.3: + dependencies: + path-key: 3.1.1 + shebang-command: 2.0.0 + which: 2.0.2 + + cssesc@3.0.0: {} + + data-uri-to-buffer@4.0.1: {} + + dayjs@1.11.13: + optional: true + + debug@4.3.7: + dependencies: + ms: 2.1.3 + + deep-extend@0.6.0: {} + + deep-is@0.1.4: {} + + deepmerge@4.3.1: {} + + defaults@1.0.4: + dependencies: + clone: 1.0.4 + + dependency-tree@11.0.1: + dependencies: + commander: 12.1.0 + filing-cabinet: 5.0.2 + precinct: 12.1.2 + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + dequal@2.0.3: {} + + detect-indent@6.1.0: {} + + detective-amd@6.0.0: + dependencies: + ast-module-types: 6.0.0 + escodegen: 2.1.0 + get-amd-module-type: 6.0.0 + node-source-walk: 7.0.0 + + detective-cjs@6.0.0: + dependencies: + ast-module-types: 6.0.0 + node-source-walk: 7.0.0 + + detective-es6@5.0.0: + dependencies: + node-source-walk: 7.0.0 + + detective-postcss@7.0.0(postcss@8.4.47): + dependencies: + is-url: 1.2.4 + postcss: 8.4.47 + postcss-values-parser: 6.0.2(postcss@8.4.47) + + detective-sass@6.0.0: + dependencies: + gonzales-pe: 4.3.0 + node-source-walk: 7.0.0 + + detective-scss@5.0.0: + dependencies: + gonzales-pe: 4.3.0 + node-source-walk: 7.0.0 + + detective-stylus@5.0.0: {} + + detective-typescript@13.0.0(typescript@5.6.3): + dependencies: + '@typescript-eslint/typescript-estree': 7.18.0(typescript@5.6.3) + ast-module-types: 6.0.0 + node-source-walk: 7.0.0 + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + detective-vue2@2.1.0(typescript@5.6.3): + dependencies: + '@dependents/detective-less': 5.0.0 + '@vue/compiler-sfc': 3.5.12 + detective-es6: 5.0.0 + detective-sass: 6.0.0 + detective-scss: 5.0.0 + detective-stylus: 5.0.0 + detective-typescript: 13.0.0(typescript@5.6.3) + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + devalue@5.1.1: {} + + devlop@1.1.0: + dependencies: + dequal: 2.0.3 + + didyoumean@1.2.2: {} + + dir-glob@3.0.1: + dependencies: + path-type: 4.0.0 + + dlv@1.1.3: {} + + doctrine@3.0.0: + dependencies: + esutils: 2.0.3 + + dotenv@16.0.3: {} + + eastasianwidth@0.2.0: {} + + effect@3.10.8: + dependencies: + fast-check: 3.23.0 + optional: true + + electron-to-chromium@1.5.50: {} + + embla-carousel-reactive-utils@8.3.1(embla-carousel@8.3.1): + dependencies: + embla-carousel: 8.3.1 + + embla-carousel-svelte@8.3.1(svelte@5.1.9): + dependencies: + embla-carousel: 8.3.1 + embla-carousel-reactive-utils: 8.3.1(embla-carousel@8.3.1) + svelte: 5.1.9 + + embla-carousel@8.3.1: {} + + emoji-regex@8.0.0: {} + + emoji-regex@9.2.2: {} + + enhanced-resolve@5.17.1: + dependencies: + graceful-fs: 4.2.11 + tapable: 2.2.1 + + enquirer@2.4.1: + dependencies: + ansi-colors: 4.1.3 + strip-ansi: 6.0.1 + + entities@4.5.0: {} + + esbuild-runner@2.2.2(esbuild@0.24.0): + dependencies: + esbuild: 0.24.0 + source-map-support: 0.5.21 + tslib: 2.4.0 + optional: true + + esbuild@0.21.5: + 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 + + esbuild@0.24.0: + optionalDependencies: + '@esbuild/aix-ppc64': 0.24.0 + '@esbuild/android-arm': 0.24.0 + '@esbuild/android-arm64': 0.24.0 + '@esbuild/android-x64': 0.24.0 + '@esbuild/darwin-arm64': 0.24.0 + '@esbuild/darwin-x64': 0.24.0 + '@esbuild/freebsd-arm64': 0.24.0 + '@esbuild/freebsd-x64': 0.24.0 + '@esbuild/linux-arm': 0.24.0 + '@esbuild/linux-arm64': 0.24.0 + '@esbuild/linux-ia32': 0.24.0 + '@esbuild/linux-loong64': 0.24.0 + '@esbuild/linux-mips64el': 0.24.0 + '@esbuild/linux-ppc64': 0.24.0 + '@esbuild/linux-riscv64': 0.24.0 + '@esbuild/linux-s390x': 0.24.0 + '@esbuild/linux-x64': 0.24.0 + '@esbuild/netbsd-x64': 0.24.0 + '@esbuild/openbsd-arm64': 0.24.0 + '@esbuild/openbsd-x64': 0.24.0 + '@esbuild/sunos-x64': 0.24.0 + '@esbuild/win32-arm64': 0.24.0 + '@esbuild/win32-ia32': 0.24.0 + '@esbuild/win32-x64': 0.24.0 + optional: true + + escalade@3.2.0: {} + + escape-string-regexp@4.0.0: {} + + escodegen@2.1.0: + dependencies: + esprima: 4.0.1 + estraverse: 5.3.0 + esutils: 2.0.3 + optionalDependencies: + source-map: 0.6.1 + + eslint-compat-utils@0.5.1(eslint@8.57.1): + dependencies: + eslint: 8.57.1 + semver: 7.6.3 + + eslint-config-prettier@9.1.0(eslint@8.57.1): + dependencies: + eslint: 8.57.1 + + eslint-config-turbo@2.2.3(eslint@8.57.1): + dependencies: + eslint: 8.57.1 + eslint-plugin-turbo: 2.2.3(eslint@8.57.1) + + eslint-plugin-svelte@2.46.0(eslint@8.57.1)(svelte@5.1.9): + dependencies: + '@eslint-community/eslint-utils': 4.4.1(eslint@8.57.1) + '@jridgewell/sourcemap-codec': 1.5.0 + eslint: 8.57.1 + eslint-compat-utils: 0.5.1(eslint@8.57.1) + esutils: 2.0.3 + known-css-properties: 0.35.0 + postcss: 8.4.47 + postcss-load-config: 3.1.4(postcss@8.4.47) + postcss-safe-parser: 6.0.0(postcss@8.4.47) + postcss-selector-parser: 6.1.2 + semver: 7.6.3 + svelte-eslint-parser: 0.43.0(svelte@5.1.9) + optionalDependencies: + svelte: 5.1.9 + transitivePeerDependencies: + - ts-node + + eslint-plugin-turbo@2.2.3(eslint@8.57.1): + dependencies: + dotenv: 16.0.3 + eslint: 8.57.1 + + eslint-scope@7.2.2: + dependencies: + esrecurse: 4.3.0 + estraverse: 5.3.0 + + eslint-visitor-keys@3.4.3: {} + + eslint@8.57.1: + dependencies: + '@eslint-community/eslint-utils': 4.4.1(eslint@8.57.1) + '@eslint-community/regexpp': 4.12.1 + '@eslint/eslintrc': 2.1.4 + '@eslint/js': 8.57.1 + '@humanwhocodes/config-array': 0.13.0 + '@humanwhocodes/module-importer': 1.0.1 + '@nodelib/fs.walk': 1.2.8 + '@ungap/structured-clone': 1.2.0 + ajv: 6.12.6 + chalk: 4.1.2 + cross-spawn: 7.0.3 + debug: 4.3.7 + doctrine: 3.0.0 + escape-string-regexp: 4.0.0 + eslint-scope: 7.2.2 + eslint-visitor-keys: 3.4.3 + espree: 9.6.1 + esquery: 1.6.0 + esutils: 2.0.3 + fast-deep-equal: 3.1.3 + file-entry-cache: 6.0.1 + find-up: 5.0.0 + glob-parent: 6.0.2 + globals: 13.24.0 + graphemer: 1.4.0 + ignore: 5.3.2 + imurmurhash: 0.1.4 + is-glob: 4.0.3 + is-path-inside: 3.0.3 + js-yaml: 4.1.0 + json-stable-stringify-without-jsonify: 1.0.1 + levn: 0.4.1 + lodash.merge: 4.6.2 + minimatch: 3.1.2 + natural-compare: 1.4.0 + optionator: 0.9.4 + strip-ansi: 6.0.1 + text-table: 0.2.0 + transitivePeerDependencies: + - supports-color + + esm-env@1.1.4: {} + + espree@9.6.1: + dependencies: + acorn: 8.12.1 + acorn-jsx: 5.3.2(acorn@8.12.1) + eslint-visitor-keys: 3.4.3 + + esprima@4.0.1: {} + + esquery@1.6.0: + dependencies: + estraverse: 5.3.0 + + esrap@1.2.2: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.0 + '@types/estree': 1.0.6 + + esrecurse@4.3.0: + dependencies: + estraverse: 5.3.0 + + estraverse@5.3.0: {} + + estree-walker@2.0.2: {} + + esutils@2.0.3: {} + + extendable-error@0.1.7: {} + + external-editor@3.1.0: + dependencies: + chardet: 0.7.0 + iconv-lite: 0.4.24 + tmp: 0.0.33 + + fast-check@3.23.0: + dependencies: + pure-rand: 6.1.0 + optional: true + + fast-deep-equal@3.1.3: {} + + fast-glob@3.3.2: + dependencies: + '@nodelib/fs.stat': 2.0.5 + '@nodelib/fs.walk': 1.2.8 + glob-parent: 5.1.2 + merge2: 1.4.1 + micromatch: 4.0.8 + + fast-json-stable-stringify@2.1.0: {} + + fast-levenshtein@2.0.6: {} + + fast-xml-parser@4.4.1: + dependencies: + strnum: 1.0.5 + + fastq@1.17.1: + dependencies: + reusify: 1.0.4 + + fdir@6.4.2(picomatch@4.0.2): + optionalDependencies: + picomatch: 4.0.2 + + fetch-blob@3.2.0: + dependencies: + node-domexception: 1.0.0 + web-streams-polyfill: 3.3.3 + + file-entry-cache@6.0.1: + dependencies: + flat-cache: 3.2.0 + + filing-cabinet@5.0.2: + dependencies: + app-module-path: 2.2.0 + commander: 12.1.0 + enhanced-resolve: 5.17.1 + module-definition: 6.0.0 + module-lookup-amd: 9.0.2 + resolve: 1.22.8 + resolve-dependency-path: 4.0.0 + sass-lookup: 6.0.1 + stylus-lookup: 6.0.0 + tsconfig-paths: 4.2.0 + typescript: 5.6.3 + + fill-range@7.1.1: + dependencies: + to-regex-range: 5.0.1 + + find-up@4.1.0: + dependencies: + locate-path: 5.0.0 + path-exists: 4.0.0 + + find-up@5.0.0: + dependencies: + locate-path: 6.0.0 + path-exists: 4.0.0 + + flat-cache@3.2.0: + dependencies: + flatted: 3.3.1 + keyv: 4.5.4 + rimraf: 3.0.2 + + flatted@3.3.1: {} + + focus-trap@7.6.0: + dependencies: + tabbable: 6.2.0 + + foreground-child@3.3.0: + dependencies: + cross-spawn: 7.0.3 + signal-exit: 4.1.0 + + formdata-polyfill@4.0.10: + dependencies: + fetch-blob: 3.2.0 + + formsnap@1.0.1(svelte@5.1.9)(sveltekit-superforms@2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)): + dependencies: + nanoid: 5.0.8 + svelte: 5.1.9 + sveltekit-superforms: 2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3) + + fraction.js@4.3.7: {} + + fs-extra@11.2.0: + dependencies: + graceful-fs: 4.2.11 + jsonfile: 6.1.0 + universalify: 2.0.1 + + fs-extra@7.0.1: + dependencies: + graceful-fs: 4.2.11 + jsonfile: 4.0.0 + universalify: 0.1.2 + + fs-extra@8.1.0: + dependencies: + graceful-fs: 4.2.11 + jsonfile: 4.0.0 + universalify: 0.1.2 + + fs.realpath@1.0.0: {} + + fsevents@2.3.3: + optional: true + + function-bind@1.1.2: {} + + gensync@1.0.0-beta.2: {} + + get-amd-module-type@6.0.0: + dependencies: + ast-module-types: 6.0.0 + node-source-walk: 7.0.0 + + get-own-enumerable-property-symbols@3.0.2: {} + + glob-parent@5.1.2: + dependencies: + is-glob: 4.0.3 + + glob-parent@6.0.2: + dependencies: + is-glob: 4.0.3 + + glob@10.4.5: + dependencies: + foreground-child: 3.3.0 + jackspeak: 3.4.3 + minimatch: 9.0.5 + minipass: 7.1.2 + package-json-from-dist: 1.0.1 + path-scurry: 1.11.1 + + glob@7.2.3: + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 3.1.2 + once: 1.4.0 + path-is-absolute: 1.0.1 + + globals@11.12.0: {} + + globals@13.24.0: + dependencies: + type-fest: 0.20.2 + + globalyzer@0.1.0: {} + + globby@11.1.0: + dependencies: + array-union: 2.1.0 + dir-glob: 3.0.1 + fast-glob: 3.3.2 + ignore: 5.3.2 + merge2: 1.4.1 + slash: 3.0.0 + + globrex@0.1.2: {} + + gonzales-pe@4.3.0: + dependencies: + minimist: 1.2.8 + + graceful-fs@4.2.11: {} + + graphemer@1.4.0: {} + + has-flag@4.0.0: {} + + hasown@2.0.2: + dependencies: + function-bind: 1.1.2 + + hast-util-to-html@9.0.3: + dependencies: + '@types/hast': 3.0.4 + '@types/unist': 3.0.3 + ccount: 2.0.1 + comma-separated-tokens: 2.0.3 + hast-util-whitespace: 3.0.0 + html-void-elements: 3.0.0 + mdast-util-to-hast: 13.2.0 + property-information: 6.5.0 + space-separated-tokens: 2.0.2 + stringify-entities: 4.0.4 + zwitch: 2.0.4 + + hast-util-whitespace@3.0.0: + dependencies: + '@types/hast': 3.0.4 + + html-void-elements@3.0.0: {} + + https-proxy-agent@7.0.5: + dependencies: + agent-base: 7.1.1 + debug: 4.3.7 + transitivePeerDependencies: + - supports-color + + human-id@1.0.2: {} + + iconv-lite@0.4.24: + dependencies: + safer-buffer: 2.1.2 + + ieee754@1.2.1: {} + + ignore@5.3.2: {} + + import-fresh@3.3.0: + dependencies: + parent-module: 1.0.1 + resolve-from: 4.0.0 + + import-meta-resolve@4.1.0: {} + + imurmurhash@0.1.4: {} + + inflight@1.0.6: + dependencies: + once: 1.4.0 + wrappy: 1.0.2 + + inherits@2.0.4: {} + + ini@1.3.8: {} + + inline-style-parser@0.2.4: {} + + interpret@1.4.0: {} + + is-binary-path@2.1.0: + dependencies: + binary-extensions: 2.3.0 + + is-core-module@2.15.1: + dependencies: + hasown: 2.0.2 + + is-extglob@2.1.1: {} + + is-fullwidth-code-point@3.0.0: {} + + is-glob@4.0.3: + dependencies: + is-extglob: 2.1.1 + + is-interactive@1.0.0: {} + + is-number@7.0.0: {} + + is-obj@1.0.1: {} + + is-path-inside@3.0.3: {} + + is-reference@3.0.2: + dependencies: + '@types/estree': 1.0.6 + + is-regexp@1.0.0: {} + + is-subdir@1.2.0: + dependencies: + better-path-resolve: 1.0.0 + + is-unicode-supported@0.1.0: {} + + is-url-superb@4.0.0: {} + + is-url@1.2.4: {} + + is-windows@1.0.2: {} + + isexe@2.0.0: {} + + jackspeak@3.4.3: + dependencies: + '@isaacs/cliui': 8.0.2 + optionalDependencies: + '@pkgjs/parseargs': 0.11.0 + + jiti@1.21.6: {} + + joi@17.13.3: + dependencies: + '@hapi/hoek': 9.3.0 + '@hapi/topo': 5.1.0 + '@sideway/address': 4.1.5 + '@sideway/formula': 3.0.1 + '@sideway/pinpoint': 2.0.0 + optional: true + + js-tokens@4.0.0: {} + + js-yaml@3.14.1: + dependencies: + argparse: 1.0.10 + esprima: 4.0.1 + + js-yaml@4.1.0: + dependencies: + argparse: 2.0.1 + + jsesc@3.0.2: {} + + json-buffer@3.0.1: {} + + json-schema-to-ts@3.1.1: + dependencies: + '@babel/runtime': 7.26.0 + ts-algebra: 2.0.0 + optional: true + + json-schema-traverse@0.4.1: {} + + json-stable-stringify-without-jsonify@1.0.1: {} + + json5@2.2.3: {} + + jsonfile@4.0.0: + optionalDependencies: + graceful-fs: 4.2.11 + + jsonfile@6.1.0: + dependencies: + universalify: 2.0.1 + optionalDependencies: + graceful-fs: 4.2.11 + + just-clone@6.2.0: {} + + keyv@4.5.4: + dependencies: + json-buffer: 3.0.1 + + kleur@4.1.5: {} + + known-css-properties@0.35.0: {} + + levn@0.4.1: + dependencies: + prelude-ls: 1.2.1 + type-check: 0.4.0 + + libphonenumber-js@1.11.12: + optional: true + + lilconfig@2.1.0: {} + + lilconfig@3.1.2: {} + + lines-and-columns@1.2.4: {} + + linkify-it@5.0.0: + dependencies: + uc.micro: 2.1.0 + + locate-character@3.0.0: {} + + locate-path@5.0.0: + dependencies: + p-locate: 4.1.0 + + locate-path@6.0.0: + dependencies: + p-locate: 5.0.0 + + lodash.castarray@4.4.0: {} + + lodash.isplainobject@4.0.6: {} + + lodash.merge@4.6.2: {} + + lodash.startcase@4.4.0: {} + + lodash@4.17.21: {} + + log-symbols@4.1.0: + dependencies: + chalk: 4.1.2 + is-unicode-supported: 0.1.0 + + lru-cache@10.4.3: {} + + lru-cache@4.1.5: + dependencies: + pseudomap: 1.0.2 + yallist: 2.1.2 + + lru-cache@5.1.1: + dependencies: + yallist: 3.1.1 + + lucide-svelte@0.454.0(svelte@5.1.9): + dependencies: + svelte: 5.1.9 + + lunr@2.3.9: {} + + madge@8.0.0(typescript@5.5.4): + dependencies: + chalk: 4.1.2 + commander: 7.2.0 + commondir: 1.0.1 + debug: 4.3.7 + dependency-tree: 11.0.1 + ora: 5.4.1 + pluralize: 8.0.0 + pretty-ms: 7.0.1 + rc: 1.2.8 + stream-to-array: 2.3.0 + ts-graphviz: 2.1.4 + walkdir: 0.4.1 + optionalDependencies: + typescript: 5.5.4 + transitivePeerDependencies: + - supports-color + + madge@8.0.0(typescript@5.6.3): + dependencies: + chalk: 4.1.2 + commander: 7.2.0 + commondir: 1.0.1 + debug: 4.3.7 + dependency-tree: 11.0.1 + ora: 5.4.1 + pluralize: 8.0.0 + pretty-ms: 7.0.1 + rc: 1.2.8 + stream-to-array: 2.3.0 + ts-graphviz: 2.1.4 + walkdir: 0.4.1 + optionalDependencies: + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + magic-string@0.30.12: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.0 + + markdown-it@14.1.0: + dependencies: + argparse: 2.0.1 + entities: 4.5.0 + linkify-it: 5.0.0 + mdurl: 2.0.0 + punycode.js: 2.3.1 + uc.micro: 2.1.0 + + mdast-util-to-hast@13.2.0: + dependencies: + '@types/hast': 3.0.4 + '@types/mdast': 4.0.4 + '@ungap/structured-clone': 1.2.0 + devlop: 1.1.0 + micromark-util-sanitize-uri: 2.0.0 + trim-lines: 3.0.1 + unist-util-position: 5.0.0 + unist-util-visit: 5.0.0 + vfile: 6.0.3 + + mdurl@2.0.0: {} + + memoize-weak@1.0.2: {} + + merge2@1.4.1: {} + + micromark-util-character@2.1.0: + dependencies: + micromark-util-symbol: 2.0.0 + micromark-util-types: 2.0.0 + + micromark-util-encode@2.0.0: {} + + micromark-util-sanitize-uri@2.0.0: + dependencies: + micromark-util-character: 2.1.0 + micromark-util-encode: 2.0.0 + micromark-util-symbol: 2.0.0 + + micromark-util-symbol@2.0.0: {} + + micromark-util-types@2.0.0: {} + + micromatch@4.0.8: + dependencies: + braces: 3.0.3 + picomatch: 2.3.1 + + mimic-fn@2.1.0: {} + + mini-svg-data-uri@1.4.4: {} + + minimatch@10.0.1: + dependencies: + brace-expansion: 2.0.1 + + minimatch@3.1.2: + dependencies: + brace-expansion: 1.1.11 + + minimatch@9.0.5: + dependencies: + brace-expansion: 2.0.1 + + minimist@1.2.8: {} + + minipass@7.1.2: {} + + minizlib@3.0.1: + dependencies: + minipass: 7.1.2 + rimraf: 5.0.10 + + mkdirp@3.0.1: {} + + mode-watcher@0.4.1(svelte@5.1.9): + dependencies: + svelte: 5.1.9 + + module-definition@6.0.0: + dependencies: + ast-module-types: 6.0.0 + node-source-walk: 7.0.0 + + module-lookup-amd@9.0.2: + dependencies: + commander: 12.1.0 + glob: 7.2.3 + requirejs: 2.3.7 + requirejs-config-file: 4.0.0 + + mri@1.2.0: {} + + mrmime@2.0.0: {} + + ms@2.1.3: {} + + mylas@2.1.13: {} + + mz@2.7.0: + dependencies: + any-promise: 1.3.0 + object-assign: 4.1.1 + thenify-all: 1.6.0 + + nanoid@3.3.7: {} + + nanoid@5.0.8: {} + + natural-compare@1.4.0: {} + + node-domexception@1.0.0: {} + + node-fetch@3.3.2: + dependencies: + data-uri-to-buffer: 4.0.1 + fetch-blob: 3.2.0 + formdata-polyfill: 4.0.10 + + node-releases@2.0.18: {} + + node-source-walk@7.0.0: + dependencies: + '@babel/parser': 7.26.2 + + normalize-path@3.0.0: {} + + normalize-range@0.1.2: {} + + normalize-url@8.0.1: + optional: true + + npm-normalize-package-bin@4.0.0: {} + + object-assign@4.1.1: {} + + object-hash@3.0.0: {} + + once@1.4.0: + dependencies: + wrappy: 1.0.2 + + onetime@5.1.2: + dependencies: + mimic-fn: 2.1.0 + + oniguruma-to-js@0.4.3: + dependencies: + regex: 4.4.0 + + optionator@0.9.4: + dependencies: + deep-is: 0.1.4 + fast-levenshtein: 2.0.6 + levn: 0.4.1 + prelude-ls: 1.2.1 + type-check: 0.4.0 + word-wrap: 1.2.5 + + ora@5.4.1: + dependencies: + bl: 4.1.0 + chalk: 4.1.2 + cli-cursor: 3.1.0 + cli-spinners: 2.9.2 + is-interactive: 1.0.0 + is-unicode-supported: 0.1.0 + log-symbols: 4.1.0 + strip-ansi: 6.0.1 + wcwidth: 1.0.1 + + os-tmpdir@1.0.2: {} + + outdent@0.5.0: {} + + p-filter@2.1.0: + dependencies: + p-map: 2.1.0 + + p-limit@2.3.0: + dependencies: + p-try: 2.2.0 + + p-limit@3.1.0: + dependencies: + yocto-queue: 0.1.0 + + p-locate@4.1.0: + dependencies: + p-limit: 2.3.0 + + p-locate@5.0.0: + dependencies: + p-limit: 3.1.0 + + p-map@2.1.0: {} + + p-try@2.2.0: {} + + package-json-from-dist@1.0.1: {} + + package-manager-detector@0.2.2: {} + + paneforge@1.0.0-next.1(svelte@5.1.9): + dependencies: + svelte: 5.1.9 + svelte-toolbelt: 0.4.6(svelte@5.1.9) + + parent-module@1.0.1: + dependencies: + callsites: 3.1.0 + + parse-ms@2.1.0: {} + + path-exists@4.0.0: {} + + path-is-absolute@1.0.1: {} + + path-key@3.1.1: {} + + path-parse@1.0.7: {} + + path-scurry@1.11.1: + dependencies: + lru-cache: 10.4.3 + minipass: 7.1.2 + + path-type@4.0.0: {} + + picocolors@1.1.0: {} + + picocolors@1.1.1: {} + + picomatch@2.3.1: {} + + picomatch@4.0.2: {} + + pify@2.3.0: {} + + pify@4.0.1: {} + + pirates@4.0.6: {} + + plimit-lit@1.6.1: + dependencies: + queue-lit: 1.5.2 + + pluralize@8.0.0: {} + + postcss-import@15.1.0(postcss@8.4.47): + dependencies: + postcss: 8.4.47 + postcss-value-parser: 4.2.0 + read-cache: 1.0.0 + resolve: 1.22.8 + + postcss-js@4.0.1(postcss@8.4.47): + dependencies: + camelcase-css: 2.0.1 + postcss: 8.4.47 + + postcss-load-config@3.1.4(postcss@8.4.47): + dependencies: + lilconfig: 2.1.0 + yaml: 1.10.2 + optionalDependencies: + postcss: 8.4.47 + + postcss-load-config@4.0.2(postcss@8.4.47): + dependencies: + lilconfig: 3.1.2 + yaml: 2.6.0 + optionalDependencies: + postcss: 8.4.47 + + postcss-nested@6.2.0(postcss@8.4.47): + dependencies: + postcss: 8.4.47 + postcss-selector-parser: 6.1.2 + + postcss-safe-parser@6.0.0(postcss@8.4.47): + dependencies: + postcss: 8.4.47 + + postcss-scss@4.0.9(postcss@8.4.47): + dependencies: + postcss: 8.4.47 + + postcss-selector-parser@6.0.10: + dependencies: + cssesc: 3.0.0 + util-deprecate: 1.0.2 + + postcss-selector-parser@6.1.2: + dependencies: + cssesc: 3.0.0 + util-deprecate: 1.0.2 + + postcss-value-parser@4.2.0: {} + + postcss-values-parser@6.0.2(postcss@8.4.47): + dependencies: + color-name: 1.1.4 + is-url-superb: 4.0.0 + postcss: 8.4.47 + quote-unquote: 1.0.0 + + postcss@8.4.47: + dependencies: + nanoid: 3.3.7 + picocolors: 1.1.0 + source-map-js: 1.2.1 + + precinct@12.1.2: + dependencies: + '@dependents/detective-less': 5.0.0 + commander: 12.1.0 + detective-amd: 6.0.0 + detective-cjs: 6.0.0 + detective-es6: 5.0.0 + detective-postcss: 7.0.0(postcss@8.4.47) + detective-sass: 6.0.0 + detective-scss: 5.0.0 + detective-stylus: 5.0.0 + detective-typescript: 13.0.0(typescript@5.6.3) + detective-vue2: 2.1.0(typescript@5.6.3) + module-definition: 6.0.0 + node-source-walk: 7.0.0 + postcss: 8.4.47 + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + + prelude-ls@1.2.1: {} + + prettier-plugin-svelte@3.2.7(prettier@3.3.3)(svelte@5.1.9): + dependencies: + prettier: 3.3.3 + svelte: 5.1.9 + + prettier-plugin-tailwindcss@0.6.8(@ianvs/prettier-plugin-sort-imports@4.3.1(@vue/compiler-sfc@3.5.12)(prettier@3.3.3))(prettier-plugin-svelte@3.2.7(prettier@3.3.3)(svelte@5.1.9))(prettier@3.3.3): + dependencies: + prettier: 3.3.3 + optionalDependencies: + '@ianvs/prettier-plugin-sort-imports': 4.3.1(@vue/compiler-sfc@3.5.12)(prettier@3.3.3) + prettier-plugin-svelte: 3.2.7(prettier@3.3.3)(svelte@5.1.9) + + prettier@2.8.8: {} + + prettier@3.3.3: {} + + pretty-ms@7.0.1: + dependencies: + parse-ms: 2.1.0 + + proc-log@5.0.0: {} + + property-expr@2.0.6: + optional: true + + property-information@6.5.0: {} + + pseudomap@1.0.2: {} + + punycode.js@2.3.1: {} + + punycode@2.3.1: {} + + pure-rand@6.1.0: + optional: true + + queue-lit@1.5.2: {} + + queue-microtask@1.2.3: {} + + quote-unquote@1.0.0: {} + + rc@1.2.8: + dependencies: + deep-extend: 0.6.0 + ini: 1.3.8 + minimist: 1.2.8 + strip-json-comments: 2.0.1 + + read-cache@1.0.0: + dependencies: + pify: 2.3.0 + + read-cmd-shim@5.0.0: {} + + read-yaml-file@1.1.0: + dependencies: + graceful-fs: 4.2.11 + js-yaml: 3.14.1 + pify: 4.0.1 + strip-bom: 3.0.0 + + readable-stream@3.6.2: + dependencies: + inherits: 2.0.4 + string_decoder: 1.3.0 + util-deprecate: 1.0.2 + + readdirp@3.6.0: + dependencies: + picomatch: 2.3.1 + + readdirp@4.0.2: {} + + rechoir@0.6.2: + dependencies: + resolve: 1.22.8 + + regenerator-runtime@0.14.1: {} + + regex@4.4.0: {} + + requirejs-config-file@4.0.0: + dependencies: + esprima: 4.0.1 + stringify-object: 3.3.0 + + requirejs@2.3.7: {} + + resolve-dependency-path@4.0.0: {} + + resolve-from@4.0.0: {} + + resolve-from@5.0.0: {} + + resolve@1.22.8: + dependencies: + is-core-module: 2.15.1 + path-parse: 1.0.7 + supports-preserve-symlinks-flag: 1.0.0 + + restore-cursor@3.1.0: + dependencies: + onetime: 5.1.2 + signal-exit: 3.0.7 + + reusify@1.0.4: {} + + rimraf@3.0.2: + dependencies: + glob: 7.2.3 + + rimraf@5.0.10: + dependencies: + glob: 10.4.5 + + rollup@4.24.3: + dependencies: + '@types/estree': 1.0.6 + optionalDependencies: + '@rollup/rollup-android-arm-eabi': 4.24.3 + '@rollup/rollup-android-arm64': 4.24.3 + '@rollup/rollup-darwin-arm64': 4.24.3 + '@rollup/rollup-darwin-x64': 4.24.3 + '@rollup/rollup-freebsd-arm64': 4.24.3 + '@rollup/rollup-freebsd-x64': 4.24.3 + '@rollup/rollup-linux-arm-gnueabihf': 4.24.3 + '@rollup/rollup-linux-arm-musleabihf': 4.24.3 + '@rollup/rollup-linux-arm64-gnu': 4.24.3 + '@rollup/rollup-linux-arm64-musl': 4.24.3 + '@rollup/rollup-linux-powerpc64le-gnu': 4.24.3 + '@rollup/rollup-linux-riscv64-gnu': 4.24.3 + '@rollup/rollup-linux-s390x-gnu': 4.24.3 + '@rollup/rollup-linux-x64-gnu': 4.24.3 + '@rollup/rollup-linux-x64-musl': 4.24.3 + '@rollup/rollup-win32-arm64-msvc': 4.24.3 + '@rollup/rollup-win32-ia32-msvc': 4.24.3 + '@rollup/rollup-win32-x64-msvc': 4.24.3 + fsevents: 2.3.3 + + run-parallel@1.2.0: + dependencies: + queue-microtask: 1.2.3 + + runed@0.15.3(svelte@5.1.9): + dependencies: + esm-env: 1.1.4 + svelte: 5.1.9 + + sade@1.8.1: + dependencies: + mri: 1.2.0 + + safe-buffer@5.2.1: {} + + safer-buffer@2.1.2: {} + + sass-lookup@6.0.1: + dependencies: + commander: 12.1.0 + + semver@6.3.1: {} + + semver@7.6.3: {} + + set-cookie-parser@2.7.1: {} + + shebang-command@1.2.0: + dependencies: + shebang-regex: 1.0.0 + + shebang-command@2.0.0: + dependencies: + shebang-regex: 3.0.0 + + shebang-regex@1.0.0: {} + + shebang-regex@3.0.0: {} + + shelljs@0.8.5: + dependencies: + glob: 7.2.3 + interpret: 1.4.0 + rechoir: 0.6.2 + + shiki@1.22.2: + dependencies: + '@shikijs/core': 1.22.2 + '@shikijs/engine-javascript': 1.22.2 + '@shikijs/engine-oniguruma': 1.22.2 + '@shikijs/types': 1.22.2 + '@shikijs/vscode-textmate': 9.3.0 + '@types/hast': 3.0.4 + + shx@0.3.4: + dependencies: + minimist: 1.2.8 + shelljs: 0.8.5 + + signal-exit@3.0.7: {} + + signal-exit@4.1.0: {} + + sirv@3.0.0: + dependencies: + '@polka/url': 1.0.0-next.28 + mrmime: 2.0.0 + totalist: 3.0.1 + + slash@3.0.0: {} + + source-map-js@1.2.1: {} + + source-map-support@0.5.21: + dependencies: + buffer-from: 1.1.2 + source-map: 0.6.1 + optional: true + + source-map@0.6.1: + optional: true + + space-separated-tokens@2.0.2: {} + + spawndamnit@2.0.0: + dependencies: + cross-spawn: 5.1.0 + signal-exit: 3.0.7 + + sprintf-js@1.0.3: {} + + stream-to-array@2.3.0: + dependencies: + any-promise: 1.3.0 + + string-width@4.2.3: + dependencies: + emoji-regex: 8.0.0 + is-fullwidth-code-point: 3.0.0 + strip-ansi: 6.0.1 + + string-width@5.1.2: + dependencies: + eastasianwidth: 0.2.0 + emoji-regex: 9.2.2 + strip-ansi: 7.1.0 + + string_decoder@1.3.0: + dependencies: + safe-buffer: 5.2.1 + + stringify-entities@4.0.4: + dependencies: + character-entities-html4: 2.1.0 + character-entities-legacy: 3.0.0 + + stringify-object@3.3.0: + dependencies: + get-own-enumerable-property-symbols: 3.0.2 + is-obj: 1.0.1 + is-regexp: 1.0.0 + + strip-ansi@6.0.1: + dependencies: + ansi-regex: 5.0.1 + + strip-ansi@7.1.0: + dependencies: + ansi-regex: 6.1.0 + + strip-bom@3.0.0: {} + + strip-json-comments@2.0.1: {} + + strip-json-comments@3.1.1: {} + + strnum@1.0.5: {} + + style-to-object@1.0.8: + dependencies: + inline-style-parser: 0.2.4 + + stylus-lookup@6.0.0: + dependencies: + commander: 12.1.0 + + sucrase@3.35.0: + dependencies: + '@jridgewell/gen-mapping': 0.3.5 + commander: 4.1.1 + glob: 10.4.5 + lines-and-columns: 1.2.4 + mz: 2.7.0 + pirates: 4.0.6 + ts-interface-checker: 0.1.13 + + supabase@1.207.9: + dependencies: + bin-links: 5.0.0 + https-proxy-agent: 7.0.5 + node-fetch: 3.3.2 + tar: 7.4.3 + transitivePeerDependencies: + - supports-color + + superstruct@2.0.2: + optional: true + + supports-color@7.2.0: + dependencies: + has-flag: 4.0.0 + + supports-preserve-symlinks-flag@1.0.0: {} + + svelte-check@4.0.5(picomatch@4.0.2)(svelte@5.1.9)(typescript@5.5.4): + dependencies: + '@jridgewell/trace-mapping': 0.3.25 + chokidar: 4.0.1 + fdir: 6.4.2(picomatch@4.0.2) + picocolors: 1.1.0 + sade: 1.8.1 + svelte: 5.1.9 + typescript: 5.5.4 + transitivePeerDependencies: + - picomatch + + svelte-eslint-parser@0.43.0(svelte@5.1.9): + dependencies: + eslint-scope: 7.2.2 + eslint-visitor-keys: 3.4.3 + espree: 9.6.1 + postcss: 8.4.47 + postcss-scss: 4.0.9(postcss@8.4.47) + optionalDependencies: + svelte: 5.1.9 + + svelte-persisted-store@0.11.0(svelte@5.1.9): + dependencies: + svelte: 5.1.9 + + svelte-radix@2.0.1(svelte@5.1.9): + dependencies: + svelte: 5.1.9 + + svelte-sonner@0.3.28(svelte@5.1.9): + dependencies: + svelte: 5.1.9 + + svelte-toolbelt@0.4.6(svelte@5.1.9): + dependencies: + clsx: 2.1.1 + style-to-object: 1.0.8 + svelte: 5.1.9 + + svelte@5.1.9: + dependencies: + '@ampproject/remapping': 2.3.0 + '@jridgewell/sourcemap-codec': 1.5.0 + '@types/estree': 1.0.6 + acorn: 8.12.1 + acorn-typescript: 1.4.13(acorn@8.12.1) + aria-query: 5.3.2 + axobject-query: 4.1.0 + esm-env: 1.1.4 + esrap: 1.2.2 + is-reference: 3.0.2 + locate-character: 3.0.0 + magic-string: 0.30.12 + zimmerframe: 1.1.2 + + sveltekit-superforms@2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3): + dependencies: + '@sveltejs/kit': 2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)) + devalue: 5.1.1 + just-clone: 6.2.0 + memoize-weak: 1.0.2 + svelte: 5.1.9 + ts-deepmerge: 7.0.1 + optionalDependencies: + '@effect/schema': 0.75.5(effect@3.10.8) + '@exodus/schemasafe': 1.3.0 + '@gcornut/valibot-json-schema': 0.31.0 + '@sinclair/typebox': 0.32.35 + '@typeschema/class-validator': 0.3.0(@types/json-schema@7.0.15)(class-validator@0.14.1) + '@vinejs/vine': 1.8.0 + arktype: 2.0.0-rc.8 + class-validator: 0.14.1 + effect: 3.10.8 + joi: 17.13.3 + json-schema-to-ts: 3.1.1 + superstruct: 2.0.2 + valibot: 0.41.0(typescript@5.6.3) + yup: 1.4.0 + zod: 3.23.8 + zod-to-json-schema: 3.23.5(zod@3.23.8) + transitivePeerDependencies: + - '@types/json-schema' + - typescript + + tabbable@6.2.0: {} + + tailwind-merge@2.5.4: {} + + tailwind-variants@0.2.1(tailwindcss@3.4.14): + dependencies: + tailwind-merge: 2.5.4 + tailwindcss: 3.4.14 + + tailwindcss-animate@1.0.7(tailwindcss@3.4.14): + dependencies: + tailwindcss: 3.4.14 + + tailwindcss@3.4.14: + dependencies: + '@alloc/quick-lru': 5.2.0 + arg: 5.0.2 + chokidar: 3.6.0 + didyoumean: 1.2.2 + dlv: 1.1.3 + fast-glob: 3.3.2 + glob-parent: 6.0.2 + is-glob: 4.0.3 + jiti: 1.21.6 + lilconfig: 2.1.0 + micromatch: 4.0.8 + normalize-path: 3.0.0 + object-hash: 3.0.0 + picocolors: 1.1.0 + postcss: 8.4.47 + postcss-import: 15.1.0(postcss@8.4.47) + postcss-js: 4.0.1(postcss@8.4.47) + postcss-load-config: 4.0.2(postcss@8.4.47) + postcss-nested: 6.2.0(postcss@8.4.47) + postcss-selector-parser: 6.1.2 + resolve: 1.22.8 + sucrase: 3.35.0 + transitivePeerDependencies: + - ts-node + + tapable@2.2.1: {} + + tar@7.4.3: + dependencies: + '@isaacs/fs-minipass': 4.0.1 + chownr: 3.0.0 + minipass: 7.1.2 + minizlib: 3.0.1 + mkdirp: 3.0.1 + yallist: 5.0.0 + + tauri-api-adapter@0.3.8(tslib@2.8.1)(typescript@5.5.4): + dependencies: + '@huakunshen/comlink': 4.4.1 + '@rollup/plugin-alias': 5.1.1(rollup@4.24.3) + '@rollup/plugin-typescript': 11.1.6(rollup@4.24.3)(tslib@2.8.1)(typescript@5.5.4) + '@tauri-apps/api': 2.0.3 + '@tauri-apps/plugin-dialog': 2.0.1 + '@tauri-apps/plugin-fs': 2.0.1 + '@tauri-apps/plugin-http': 2.0.1 + '@tauri-apps/plugin-log': 2.0.0 + '@tauri-apps/plugin-notification': 2.0.0 + '@tauri-apps/plugin-os': 2.0.0 + '@tauri-apps/plugin-shell': 2.0.1 + '@tauri-apps/plugin-upload': 2.0.0 + rimraf: 5.0.10 + rollup: 4.24.3 + shx: 0.3.4 + tauri-plugin-clipboard-api: 2.1.11(typescript@5.5.4) + tauri-plugin-network-api: 2.0.4(typescript@5.5.4) + tauri-plugin-shellx-api: 2.0.11 + tauri-plugin-system-info-api: 2.0.8(typescript@5.5.4) + tsc-alias: 1.8.10 + typescript: 5.5.4 + valibot: 0.40.0(typescript@5.5.4) + transitivePeerDependencies: + - tslib + + tauri-api-adapter@0.3.8(tslib@2.8.1)(typescript@5.6.3): + dependencies: + '@huakunshen/comlink': 4.4.1 + '@rollup/plugin-alias': 5.1.1(rollup@4.24.3) + '@rollup/plugin-typescript': 11.1.6(rollup@4.24.3)(tslib@2.8.1)(typescript@5.6.3) + '@tauri-apps/api': 2.0.3 + '@tauri-apps/plugin-dialog': 2.0.1 + '@tauri-apps/plugin-fs': 2.0.1 + '@tauri-apps/plugin-http': 2.0.1 + '@tauri-apps/plugin-log': 2.0.0 + '@tauri-apps/plugin-notification': 2.0.0 + '@tauri-apps/plugin-os': 2.0.0 + '@tauri-apps/plugin-shell': 2.0.1 + '@tauri-apps/plugin-upload': 2.0.0 + rimraf: 5.0.10 + rollup: 4.24.3 + shx: 0.3.4 + tauri-plugin-clipboard-api: 2.1.11(typescript@5.6.3) + tauri-plugin-network-api: 2.0.4(typescript@5.6.3) + tauri-plugin-shellx-api: 2.0.11 + tauri-plugin-system-info-api: 2.0.8(typescript@5.6.3) + tsc-alias: 1.8.10 + typescript: 5.6.3 + valibot: 0.40.0(typescript@5.6.3) + transitivePeerDependencies: + - tslib + + tauri-plugin-clipboard-api@2.1.11(typescript@5.5.4): + dependencies: + '@tauri-apps/api': 2.0.1 + valibot: 0.40.0(typescript@5.5.4) + transitivePeerDependencies: + - typescript + + tauri-plugin-clipboard-api@2.1.11(typescript@5.6.3): + dependencies: + '@tauri-apps/api': 2.0.1 + valibot: 0.40.0(typescript@5.6.3) + transitivePeerDependencies: + - typescript + + tauri-plugin-network-api@2.0.4(typescript@5.5.4): + dependencies: + '@tauri-apps/api': 2.0.3 + valibot: 0.40.0(typescript@5.5.4) + transitivePeerDependencies: + - typescript + + tauri-plugin-network-api@2.0.4(typescript@5.6.3): + dependencies: + '@tauri-apps/api': 2.0.3 + valibot: 0.40.0(typescript@5.6.3) + transitivePeerDependencies: + - typescript + + tauri-plugin-shellx-api@2.0.11: + dependencies: + '@tauri-apps/api': 2.0.3 + + tauri-plugin-system-info-api@2.0.8(typescript@5.5.4): + dependencies: + '@tauri-apps/api': 2.0.3 + valibot: 0.40.0(typescript@5.5.4) + transitivePeerDependencies: + - typescript + + tauri-plugin-system-info-api@2.0.8(typescript@5.6.3): + dependencies: + '@tauri-apps/api': 2.0.3 + valibot: 0.40.0(typescript@5.6.3) + transitivePeerDependencies: + - typescript + + term-size@2.2.1: {} + + terser@5.36.0: + dependencies: + '@jridgewell/source-map': 0.3.6 + acorn: 8.12.1 + commander: 2.20.3 + source-map-support: 0.5.21 + optional: true + + text-table@0.2.0: {} + + thenify-all@1.6.0: + dependencies: + thenify: 3.3.1 + + thenify@3.3.1: + dependencies: + any-promise: 1.3.0 + + tiny-case@1.0.3: + optional: true + + tiny-glob@0.2.9: + dependencies: + globalyzer: 0.1.0 + globrex: 0.1.2 + + tmp@0.0.33: + dependencies: + os-tmpdir: 1.0.2 + + to-regex-range@5.0.1: + dependencies: + is-number: 7.0.0 + + toposort@2.0.2: + optional: true + + totalist@3.0.1: {} + + tr46@0.0.3: {} + + trim-lines@3.0.1: {} + + ts-algebra@2.0.0: + optional: true + + ts-api-utils@1.4.0(typescript@5.6.3): + dependencies: + typescript: 5.6.3 + + ts-deepmerge@7.0.1: {} + + ts-graphviz@2.1.4: + dependencies: + '@ts-graphviz/adapter': 2.0.5 + '@ts-graphviz/ast': 2.0.5 + '@ts-graphviz/common': 2.1.4 + '@ts-graphviz/core': 2.0.5 + + ts-interface-checker@0.1.13: {} + + tsc-alias@1.8.10: + dependencies: + chokidar: 3.6.0 + commander: 9.5.0 + globby: 11.1.0 + mylas: 2.1.13 + normalize-path: 3.0.0 + plimit-lit: 1.6.1 + + tsconfig-paths@4.2.0: + dependencies: + json5: 2.2.3 + minimist: 1.2.8 + strip-bom: 3.0.0 + + tslib@2.4.0: + optional: true + + tslib@2.8.1: {} + + turbo-darwin-64@2.2.3: + optional: true + + turbo-darwin-arm64@2.2.3: + optional: true + + turbo-linux-64@2.2.3: + optional: true + + turbo-linux-arm64@2.2.3: + optional: true + + turbo-windows-64@2.2.3: + optional: true + + turbo-windows-arm64@2.2.3: + optional: true + + turbo@2.2.3: + optionalDependencies: + turbo-darwin-64: 2.2.3 + turbo-darwin-arm64: 2.2.3 + turbo-linux-64: 2.2.3 + turbo-linux-arm64: 2.2.3 + turbo-windows-64: 2.2.3 + turbo-windows-arm64: 2.2.3 + + type-check@0.4.0: + dependencies: + prelude-ls: 1.2.1 + + type-fest@0.20.2: {} + + type-fest@2.19.0: + optional: true + + typedoc@0.26.11(typescript@5.5.4): + dependencies: + lunr: 2.3.9 + markdown-it: 14.1.0 + minimatch: 9.0.5 + shiki: 1.22.2 + typescript: 5.5.4 + yaml: 2.6.0 + + typedoc@0.26.11(typescript@5.6.3): + dependencies: + lunr: 2.3.9 + markdown-it: 14.1.0 + minimatch: 9.0.5 + shiki: 1.22.2 + typescript: 5.6.3 + yaml: 2.6.0 + + typescript@5.5.4: {} + + typescript@5.6.3: {} + + uc.micro@2.1.0: {} + + undici-types@5.26.5: {} + + undici-types@6.19.8: {} + + unist-util-is@6.0.0: + dependencies: + '@types/unist': 3.0.3 + + unist-util-position@5.0.0: + dependencies: + '@types/unist': 3.0.3 + + unist-util-stringify-position@4.0.0: + dependencies: + '@types/unist': 3.0.3 + + unist-util-visit-parents@6.0.1: + dependencies: + '@types/unist': 3.0.3 + unist-util-is: 6.0.0 + + unist-util-visit@5.0.0: + dependencies: + '@types/unist': 3.0.3 + unist-util-is: 6.0.0 + unist-util-visit-parents: 6.0.1 + + universalify@0.1.2: {} + + universalify@2.0.1: {} + + update-browserslist-db@1.1.1(browserslist@4.24.2): + dependencies: + browserslist: 4.24.2 + escalade: 3.2.0 + picocolors: 1.1.0 + + uri-js@4.4.1: + dependencies: + punycode: 2.3.1 + + util-deprecate@1.0.2: {} + + uuid@11.0.2: {} + + uuid@9.0.1: {} + + valibot@0.31.1: + optional: true + + valibot@0.40.0(typescript@5.5.4): + optionalDependencies: + typescript: 5.5.4 + + valibot@0.40.0(typescript@5.6.3): + optionalDependencies: + typescript: 5.6.3 + + valibot@0.41.0(typescript@5.6.3): + optionalDependencies: + typescript: 5.6.3 + optional: true + + valibot@0.42.1(typescript@5.5.4): + optionalDependencies: + typescript: 5.5.4 + + validator@13.12.0: + optional: true + + vaul-svelte@0.3.2(svelte@5.1.9): + dependencies: + bits-ui: 0.21.16(svelte@5.1.9) + svelte: 5.1.9 + + vfile-message@4.0.2: + dependencies: + '@types/unist': 3.0.3 + unist-util-stringify-position: 4.0.0 + + vfile@6.0.3: + dependencies: + '@types/unist': 3.0.3 + vfile-message: 4.0.2 + + vite@5.4.10(@types/node@22.8.7)(terser@5.36.0): + dependencies: + esbuild: 0.21.5 + postcss: 8.4.47 + rollup: 4.24.3 + optionalDependencies: + '@types/node': 22.8.7 + fsevents: 2.3.3 + terser: 5.36.0 + + vitefu@1.0.3(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)): + optionalDependencies: + vite: 5.4.10(@types/node@22.8.7)(terser@5.36.0) + + walkdir@0.4.1: {} + + wcwidth@1.0.1: + dependencies: + defaults: 1.0.4 + + web-streams-polyfill@3.3.3: {} + + webidl-conversions@3.0.1: {} + + whatwg-url@5.0.0: + dependencies: + tr46: 0.0.3 + webidl-conversions: 3.0.1 + + which@1.3.1: dependencies: isexe: 2.0.0 - /word-wrap@1.2.5: - resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==} - engines: {node: '>=0.10.0'} - dev: false + which@2.0.2: + dependencies: + isexe: 2.0.0 - /wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} + word-wrap@1.2.5: {} + + wrap-ansi@7.0.0: dependencies: ansi-styles: 4.3.0 string-width: 4.2.3 strip-ansi: 6.0.1 - dev: true - /wrap-ansi@8.1.0: - resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} - engines: {node: '>=12'} + wrap-ansi@8.1.0: dependencies: ansi-styles: 6.2.1 string-width: 5.1.2 strip-ansi: 7.1.0 - dev: true - /wrappy@1.0.2: - resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - dev: false + wrappy@1.0.2: {} - /yallist@3.1.1: - resolution: {integrity: sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==} - dev: true + write-file-atomic@6.0.0: + dependencies: + imurmurhash: 0.1.4 + signal-exit: 4.1.0 - /yaml@1.10.2: - resolution: {integrity: sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==} - engines: {node: '>= 6'} - dev: false + ws@8.18.0: {} - /yaml@2.6.0: - resolution: {integrity: sha512-a6ae//JvKDEra2kdi1qzCyrJW/WZCgFi8ydDV+eXExl95t+5R+ijnqHJbz9tmMh8FUjx3iv2fCQ4dclAQlO2UQ==} - engines: {node: '>= 14'} - hasBin: true - dev: true + yallist@2.1.2: {} - /yocto-queue@0.1.0: - resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} - engines: {node: '>=10'} - dev: false + yallist@3.1.1: {} - /yup@1.4.0: - resolution: {integrity: sha512-wPbgkJRCqIf+OHyiTBQoJiP5PFuAXaWiJK6AmYkzQAh5/c2K9hzSApBZG5wV9KoKSePF7sAxmNSvh/13YHkFDg==} - requiresBuild: true + yallist@5.0.0: {} + + yaml@1.10.2: {} + + yaml@2.6.0: {} + + yocto-queue@0.1.0: {} + + yup@1.4.0: dependencies: property-expr: 2.0.6 tiny-case: 1.0.3 toposort: 2.0.2 type-fest: 2.19.0 - dev: true optional: true - /zimmerframe@1.1.2: - resolution: {integrity: sha512-rAbqEGa8ovJy4pyBxZM70hg4pE6gDgaQ0Sl9M3enG3I0d6H4XSAM3GeNGLKnsBpuijUow064sf7ww1nutC5/3w==} + zimmerframe@1.1.2: {} - /zod-to-json-schema@3.23.5(zod@3.23.8): - resolution: {integrity: sha512-5wlSS0bXfF/BrL4jPAbz9da5hDlDptdEppYfe+x4eIJ7jioqKG9uUxOwPzqof09u/XeVdrgFu29lZi+8XNDJtA==} - requiresBuild: true - peerDependencies: - zod: ^3.23.3 + zod-to-json-schema@3.23.5(zod@3.23.8): dependencies: zod: 3.23.8 - dev: true optional: true - /zod@3.23.8: - resolution: {integrity: sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==} - requiresBuild: true - dev: true - optional: true + zod@3.23.8: {} + + zwitch@2.0.4: {} diff --git a/pnpm-workspace.yaml b/pnpm-workspace.yaml index 3ff5faa..1641500 100644 --- a/pnpm-workspace.yaml +++ b/pnpm-workspace.yaml @@ -1,3 +1,7 @@ packages: - "apps/*" - "packages/*" + - "packages/extensions/*" + - "packages/templates/*" + - "packages/tauri-plugins/*" + - "vendors/*" diff --git a/turbo.json b/turbo.json index 807e324..f8d7c1d 100644 --- a/turbo.json +++ b/turbo.json @@ -1,18 +1,24 @@ { - "$schema": "https://turbo.build/schema.json", - "ui": "tui", - "tasks": { - "build": { - "dependsOn": ["^build"], - "inputs": ["$TURBO_DEFAULT$", ".env*"], - "outputs": [".next/**", "!.next/cache/**"] - }, - "lint": { - "dependsOn": ["^lint"] - }, - "dev": { - "cache": false, - "persistent": true - } - } + "$schema": "https://turbo.build/schema.json", + "ui": "tui", + "tasks": { + "build": { + "dependsOn": ["^build"], + "inputs": ["$TURBO_DEFAULT$", ".env*"], + "outputs": [".next/**", "!.next/cache/**"] + }, + "lint": { + "dependsOn": ["^lint"] + }, + "test": { + "dependsOn": ["^test"] + }, + "prepare": { + "dependsOn": ["^prepare"] + }, + "dev": { + "cache": false, + "persistent": true + } + } } diff --git a/vendors/applications-rs b/vendors/applications-rs new file mode 160000 index 0000000..6b38505 --- /dev/null +++ b/vendors/applications-rs @@ -0,0 +1 @@ +Subproject commit 6b38505d8534d81d826d613527eb75fd8da4ede4 diff --git a/vendors/tauri-plugin-network b/vendors/tauri-plugin-network new file mode 160000 index 0000000..5bf470b --- /dev/null +++ b/vendors/tauri-plugin-network @@ -0,0 +1 @@ +Subproject commit 5bf470bc27207446367626d5e1a24370a3060a84 diff --git a/vendors/tauri-plugin-system-info b/vendors/tauri-plugin-system-info new file mode 160000 index 0000000..f3ec4bf --- /dev/null +++ b/vendors/tauri-plugin-system-info @@ -0,0 +1 @@ +Subproject commit f3ec4bf42e432d8f28f92966185a6e3a87cb37f3