mirror of
https://github.com/mountain-loop/yaak.git
synced 2026-03-31 06:23:08 +02:00
Compare commits
25 Commits
v2026.2.0-
...
mcp-client
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c415e7f471 | ||
|
|
8023603ebe | ||
|
|
66942eaf2c | ||
|
|
38796b1833 | ||
|
|
49ffa6fc45 | ||
|
|
1f56ba2eb6 | ||
|
|
f98a70ecb4 | ||
|
|
2984eb40c9 | ||
|
|
cc5d4742f0 | ||
|
|
5b8e4b98a0 | ||
|
|
8637c90a21 | ||
|
|
b88c5e71a0 | ||
|
|
1899d512ab | ||
|
|
7c31718f5e | ||
|
|
8f1463e5d0 | ||
|
|
0dc8807808 | ||
|
|
f24a159b8a | ||
|
|
0b91d3aaff | ||
|
|
431dc1c896 | ||
|
|
bc8277b56b | ||
|
|
0afed185d9 | ||
|
|
55cee00601 | ||
|
|
b41a8e04cb | ||
|
|
eff4519d91 | ||
|
|
c4ce458f79 |
36
Cargo.lock
generated
36
Cargo.lock
generated
@@ -689,9 +689,9 @@ checksum = "8f1fe948ff07f4bd06c30984e69f5b4899c516a3ef74f34df92a2df2ab535495"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bytes"
|
name = "bytes"
|
||||||
version = "1.10.1"
|
version = "1.11.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "d71b6127be86fdcfddb610f7182ac57211d4b18a3e9c82eb2d17662f2227ad6a"
|
checksum = "1e748733b7cbc798e1434b6ac524f0c1ff2ab456fe201501e6497c8417a4fc33"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
@@ -1316,12 +1316,12 @@ checksum = "da692b8d1080ea3045efaab14434d40468c3d8657e42abddfffca87b428f4c1b"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "deranged"
|
name = "deranged"
|
||||||
version = "0.4.0"
|
version = "0.5.5"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9c9e6a11ca8224451684bc0d7d5a7adbf8f2fd6887261a1cfc3c0432f9d4068e"
|
checksum = "ececcb659e7ba858fb4f10388c250a7252eb0a27373f1a72b8748afdd248e587"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"powerfmt",
|
"powerfmt",
|
||||||
"serde",
|
"serde_core",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -2136,9 +2136,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "git2"
|
name = "git2"
|
||||||
version = "0.20.2"
|
version = "0.20.4"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "2deb07a133b1520dc1a5690e9bd08950108873d7ed5de38dcc74d3b5ebffa110"
|
checksum = "7b88256088d75a56f8ecfa070513a775dd9107f6530ef14919dac831af9cfe2b"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bitflags 2.9.1",
|
"bitflags 2.9.1",
|
||||||
"libc",
|
"libc",
|
||||||
@@ -3036,9 +3036,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "libgit2-sys"
|
name = "libgit2-sys"
|
||||||
version = "0.18.1+1.9.0"
|
version = "0.18.3+1.9.2"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e1dcb20f84ffcdd825c7a311ae347cce604a6f084a767dec4a4929829645290e"
|
checksum = "c9b3acc4b91781bb0b3386669d325163746af5f6e4f73e6d2d630e09a35f3487"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cc",
|
"cc",
|
||||||
"libc",
|
"libc",
|
||||||
@@ -3446,9 +3446,9 @@ checksum = "5e0826a989adedc2a244799e823aece04662b66609d96af8dff7ac6df9a8925d"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "num-conv"
|
name = "num-conv"
|
||||||
version = "0.1.0"
|
version = "0.2.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "51d515d32fb182ee37cda2ccdcb92950d6a3c2893aa280e540671c2cd0f3b1d9"
|
checksum = "cf97ec579c3c42f953ef76dbf8d55ac91fb219dde70e49aa4a6b7d74e9919050"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "num-traits"
|
name = "num-traits"
|
||||||
@@ -6341,9 +6341,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "time"
|
name = "time"
|
||||||
version = "0.3.41"
|
version = "0.3.47"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8a7619e19bc266e0f9c5e6686659d394bc57973859340060a69221e57dbc0c40"
|
checksum = "743bd48c283afc0388f9b8827b976905fb217ad9e647fae3a379a9283c4def2c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"deranged",
|
"deranged",
|
||||||
"itoa",
|
"itoa",
|
||||||
@@ -6351,22 +6351,22 @@ dependencies = [
|
|||||||
"num-conv",
|
"num-conv",
|
||||||
"num_threads",
|
"num_threads",
|
||||||
"powerfmt",
|
"powerfmt",
|
||||||
"serde",
|
"serde_core",
|
||||||
"time-core",
|
"time-core",
|
||||||
"time-macros",
|
"time-macros",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "time-core"
|
name = "time-core"
|
||||||
version = "0.1.4"
|
version = "0.1.8"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c9e9a38711f559d9e3ce1cdb06dd7c5b8ea546bc90052da6d06bb76da74bb07c"
|
checksum = "7694e1cfe791f8d31026952abf09c69ca6f6fa4e1a1229e18988f06a04a12dca"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "time-macros"
|
name = "time-macros"
|
||||||
version = "0.2.22"
|
version = "0.2.27"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "3526739392ec93fd8b359c8e98514cb3e8e021beb4e5f597b00a0221f8ed8a49"
|
checksum = "2e70e4c5a0e0a8a4823ad65dfe1a6930e4f4d756dcd9dd7939022b5e8c501215"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"num-conv",
|
"num-conv",
|
||||||
"time-core",
|
"time-core",
|
||||||
|
|||||||
27
MCP_CLIENT_PLAN.md
Normal file
27
MCP_CLIENT_PLAN.md
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
# MCP Client Plan
|
||||||
|
|
||||||
|
## Goal
|
||||||
|
Add an MCP client mode to Yaak so users can connect to and debug MCP servers.
|
||||||
|
|
||||||
|
## Core Design
|
||||||
|
- **Protocol layer:** Implement JSON‑RPC framing, message IDs, and notifications as the common core.
|
||||||
|
- **Transport interface:** Define an async trait with `connect`, `send`, `receive`, and `close` methods.
|
||||||
|
- **Transports:**
|
||||||
|
- Start with **Standard I/O** for local development.
|
||||||
|
- Reuse the existing HTTP stack for **HTTP streaming** next.
|
||||||
|
- Leave hooks for **WebSocket** support later.
|
||||||
|
|
||||||
|
## Integration
|
||||||
|
- Register MCP as a new request type alongside REST, GraphQL, gRPC, and WebSocket.
|
||||||
|
- Allow per‑request transport selection (stdio or HTTP).
|
||||||
|
- Map inbound messages into a new MCP response model that feeds existing timeline and debug views.
|
||||||
|
|
||||||
|
## Testing and Dog‑fooding
|
||||||
|
- Convert Yaak's own MCP server to Standard I/O for local testing.
|
||||||
|
- Use it internally to validate protocol behavior and message flow.
|
||||||
|
- Add unit and integration tests for JSON‑RPC messaging and transport abstractions.
|
||||||
|
|
||||||
|
## Future Refinements
|
||||||
|
- Add WebSocket transport support once core paths are stable.
|
||||||
|
- Extend timelines for protocol‑level visualization layered over raw transport events.
|
||||||
|
- Implement version and capability negotiation between client and server.
|
||||||
@@ -22,7 +22,7 @@
|
|||||||
<!-- sponsors-premium --><a href="https://github.com/MVST-Solutions"><img src="https://github.com/MVST-Solutions.png" width="80px" alt="User avatar: MVST-Solutions" /></a> <a href="https://github.com/dharsanb"><img src="https://github.com/dharsanb.png" width="80px" alt="User avatar: dharsanb" /></a> <a href="https://github.com/railwayapp"><img src="https://github.com/railwayapp.png" width="80px" alt="User avatar: railwayapp" /></a> <a href="https://github.com/caseyamcl"><img src="https://github.com/caseyamcl.png" width="80px" alt="User avatar: caseyamcl" /></a> <a href="https://github.com/bytebase"><img src="https://github.com/bytebase.png" width="80px" alt="User avatar: bytebase" /></a> <a href="https://github.com/"><img src="https://raw.githubusercontent.com/JamesIves/github-sponsors-readme-action/dev/.github/assets/placeholder.png" width="80px" alt="User avatar: " /></a> <!-- sponsors-premium -->
|
<!-- sponsors-premium --><a href="https://github.com/MVST-Solutions"><img src="https://github.com/MVST-Solutions.png" width="80px" alt="User avatar: MVST-Solutions" /></a> <a href="https://github.com/dharsanb"><img src="https://github.com/dharsanb.png" width="80px" alt="User avatar: dharsanb" /></a> <a href="https://github.com/railwayapp"><img src="https://github.com/railwayapp.png" width="80px" alt="User avatar: railwayapp" /></a> <a href="https://github.com/caseyamcl"><img src="https://github.com/caseyamcl.png" width="80px" alt="User avatar: caseyamcl" /></a> <a href="https://github.com/bytebase"><img src="https://github.com/bytebase.png" width="80px" alt="User avatar: bytebase" /></a> <a href="https://github.com/"><img src="https://raw.githubusercontent.com/JamesIves/github-sponsors-readme-action/dev/.github/assets/placeholder.png" width="80px" alt="User avatar: " /></a> <!-- sponsors-premium -->
|
||||||
</p>
|
</p>
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<!-- sponsors-base --><a href="https://github.com/seanwash"><img src="https://github.com/seanwash.png" width="50px" alt="User avatar: seanwash" /></a> <a href="https://github.com/jerath"><img src="https://github.com/jerath.png" width="50px" alt="User avatar: jerath" /></a> <a href="https://github.com/itsa-sh"><img src="https://github.com/itsa-sh.png" width="50px" alt="User avatar: itsa-sh" /></a> <a href="https://github.com/dmmulroy"><img src="https://github.com/dmmulroy.png" width="50px" alt="User avatar: dmmulroy" /></a> <a href="https://github.com/timcole"><img src="https://github.com/timcole.png" width="50px" alt="User avatar: timcole" /></a> <a href="https://github.com/VLZH"><img src="https://github.com/VLZH.png" width="50px" alt="User avatar: VLZH" /></a> <a href="https://github.com/terasaka2k"><img src="https://github.com/terasaka2k.png" width="50px" alt="User avatar: terasaka2k" /></a> <a href="https://github.com/andriyor"><img src="https://github.com/andriyor.png" width="50px" alt="User avatar: andriyor" /></a> <a href="https://github.com/majudhu"><img src="https://github.com/majudhu.png" width="50px" alt="User avatar: majudhu" /></a> <a href="https://github.com/axelrindle"><img src="https://github.com/axelrindle.png" width="50px" alt="User avatar: axelrindle" /></a> <a href="https://github.com/jirizverina"><img src="https://github.com/jirizverina.png" width="50px" alt="User avatar: jirizverina" /></a> <a href="https://github.com/chip-well"><img src="https://github.com/chip-well.png" width="50px" alt="User avatar: chip-well" /></a> <a href="https://github.com/GRAYAH"><img src="https://github.com/GRAYAH.png" width="50px" alt="User avatar: GRAYAH" /></a> <!-- sponsors-base -->
|
<!-- sponsors-base --><a href="https://github.com/seanwash"><img src="https://github.com/seanwash.png" width="50px" alt="User avatar: seanwash" /></a> <a href="https://github.com/jerath"><img src="https://github.com/jerath.png" width="50px" alt="User avatar: jerath" /></a> <a href="https://github.com/itsa-sh"><img src="https://github.com/itsa-sh.png" width="50px" alt="User avatar: itsa-sh" /></a> <a href="https://github.com/dmmulroy"><img src="https://github.com/dmmulroy.png" width="50px" alt="User avatar: dmmulroy" /></a> <a href="https://github.com/timcole"><img src="https://github.com/timcole.png" width="50px" alt="User avatar: timcole" /></a> <a href="https://github.com/VLZH"><img src="https://github.com/VLZH.png" width="50px" alt="User avatar: VLZH" /></a> <a href="https://github.com/terasaka2k"><img src="https://github.com/terasaka2k.png" width="50px" alt="User avatar: terasaka2k" /></a> <a href="https://github.com/andriyor"><img src="https://github.com/andriyor.png" width="50px" alt="User avatar: andriyor" /></a> <a href="https://github.com/majudhu"><img src="https://github.com/majudhu.png" width="50px" alt="User avatar: majudhu" /></a> <a href="https://github.com/axelrindle"><img src="https://github.com/axelrindle.png" width="50px" alt="User avatar: axelrindle" /></a> <a href="https://github.com/jirizverina"><img src="https://github.com/jirizverina.png" width="50px" alt="User avatar: jirizverina" /></a> <a href="https://github.com/chip-well"><img src="https://github.com/chip-well.png" width="50px" alt="User avatar: chip-well" /></a> <a href="https://github.com/GRAYAH"><img src="https://github.com/GRAYAH.png" width="50px" alt="User avatar: GRAYAH" /></a> <a href="https://github.com/flashblaze"><img src="https://github.com/flashblaze.png" width="50px" alt="User avatar: flashblaze" /></a> <!-- sponsors-base -->
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||

|

|
||||||
|
|||||||
198
crates-cli/yaak-cli/PLAN.md
Normal file
198
crates-cli/yaak-cli/PLAN.md
Normal file
@@ -0,0 +1,198 @@
|
|||||||
|
# CLI Command Architecture Plan
|
||||||
|
|
||||||
|
## Goal
|
||||||
|
|
||||||
|
Redesign the yaak-cli command structure to use a resource-oriented `<resource> <action>`
|
||||||
|
pattern that scales well, is discoverable, and supports both human and LLM workflows.
|
||||||
|
|
||||||
|
## Command Architecture
|
||||||
|
|
||||||
|
### Design Principles
|
||||||
|
|
||||||
|
- **Resource-oriented**: top-level commands are nouns, subcommands are verbs
|
||||||
|
- **Polymorphic requests**: `request` covers HTTP, gRPC, and WebSocket — the CLI
|
||||||
|
resolves the type via `get_any_request` and adapts behavior accordingly
|
||||||
|
- **Simple creation, full-fidelity via JSON**: human-friendly flags for basic creation,
|
||||||
|
`--json` for full control (targeted at LLM and scripting workflows)
|
||||||
|
- **Runtime schema introspection**: `request schema` outputs JSON Schema for the request
|
||||||
|
models, with dynamic auth fields populated from loaded plugins at runtime
|
||||||
|
- **Destructive actions require confirmation**: `delete` commands prompt for user
|
||||||
|
confirmation before proceeding. Can be bypassed with `--yes` / `-y` for scripting
|
||||||
|
|
||||||
|
### Commands
|
||||||
|
|
||||||
|
```
|
||||||
|
# Top-level shortcut
|
||||||
|
yaakcli send <id> [-e <env_id>] # id can be a request, folder, or workspace
|
||||||
|
|
||||||
|
# Resource commands
|
||||||
|
yaakcli workspace list
|
||||||
|
yaakcli workspace show <id>
|
||||||
|
yaakcli workspace create --name <name>
|
||||||
|
yaakcli workspace create --json '{"name": "My Workspace"}'
|
||||||
|
yaakcli workspace create '{"name": "My Workspace"}' # positional JSON shorthand
|
||||||
|
yaakcli workspace update --json '{"id": "wk_abc", "name": "New Name"}'
|
||||||
|
yaakcli workspace delete <id>
|
||||||
|
|
||||||
|
yaakcli request list <workspace_id>
|
||||||
|
yaakcli request show <id>
|
||||||
|
yaakcli request create <workspace_id> --name <name> --url <url> [--method GET]
|
||||||
|
yaakcli request create --json '{"workspaceId": "wk_abc", "url": "..."}'
|
||||||
|
yaakcli request update --json '{"id": "rq_abc", "url": "https://new.com"}'
|
||||||
|
yaakcli request send <id> [-e <env_id>]
|
||||||
|
yaakcli request delete <id>
|
||||||
|
yaakcli request schema <http|grpc|websocket>
|
||||||
|
|
||||||
|
yaakcli folder list <workspace_id>
|
||||||
|
yaakcli folder show <id>
|
||||||
|
yaakcli folder create <workspace_id> --name <name>
|
||||||
|
yaakcli folder create --json '{"workspaceId": "wk_abc", "name": "Auth"}'
|
||||||
|
yaakcli folder update --json '{"id": "fl_abc", "name": "New Name"}'
|
||||||
|
yaakcli folder delete <id>
|
||||||
|
|
||||||
|
yaakcli environment list <workspace_id>
|
||||||
|
yaakcli environment show <id>
|
||||||
|
yaakcli environment create <workspace_id> --name <name>
|
||||||
|
yaakcli environment create --json '{"workspaceId": "wk_abc", "name": "Production"}'
|
||||||
|
yaakcli environment update --json '{"id": "ev_abc", ...}'
|
||||||
|
yaakcli environment delete <id>
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
### `send` — Top-Level Shortcut
|
||||||
|
|
||||||
|
`yaakcli send <id>` is a convenience alias that accepts any sendable ID. It tries
|
||||||
|
each type in order via DB lookups (short-circuiting on first match):
|
||||||
|
|
||||||
|
1. Request (HTTP, gRPC, or WebSocket via `get_any_request`)
|
||||||
|
2. Folder (sends all requests in the folder)
|
||||||
|
3. Workspace (sends all requests in the workspace)
|
||||||
|
|
||||||
|
ID prefixes exist (e.g. `rq_`, `fl_`, `wk_`) but are not relied upon — resolution
|
||||||
|
is purely by DB lookup.
|
||||||
|
|
||||||
|
`request send <id>` is the same but restricted to request IDs only.
|
||||||
|
|
||||||
|
### Request Send — Polymorphic Behavior
|
||||||
|
|
||||||
|
`send` means "execute this request" regardless of protocol:
|
||||||
|
|
||||||
|
- **HTTP**: send request, print response, exit
|
||||||
|
- **gRPC**: invoke the method; for streaming, stream output to stdout until done/Ctrl+C
|
||||||
|
- **WebSocket**: connect, stream messages to stdout until closed/Ctrl+C
|
||||||
|
|
||||||
|
### `request schema` — Runtime JSON Schema
|
||||||
|
|
||||||
|
Outputs a JSON Schema describing the full request shape, including dynamic fields:
|
||||||
|
|
||||||
|
1. Generate base schema from `schemars::JsonSchema` derive on the Rust model structs
|
||||||
|
2. Load plugins, collect auth strategy definitions and their form inputs
|
||||||
|
3. Merge plugin-defined auth fields into the `authentication` property as a `oneOf`
|
||||||
|
4. Output the combined schema as JSON
|
||||||
|
|
||||||
|
This lets an LLM call `schema`, read the shape, and construct valid JSON for
|
||||||
|
`create --json` or `update --json`.
|
||||||
|
|
||||||
|
## Implementation Steps
|
||||||
|
|
||||||
|
### Phase 1: Restructure commands (no new functionality)
|
||||||
|
|
||||||
|
Refactor `main.rs` into the new resource/action pattern using clap subcommand nesting.
|
||||||
|
Existing behavior stays the same, just reorganized. Remove the `get` command.
|
||||||
|
|
||||||
|
1. Create module structure: `commands/workspace.rs`, `commands/request.rs`, etc.
|
||||||
|
2. Define nested clap enums:
|
||||||
|
```rust
|
||||||
|
enum Commands {
|
||||||
|
Send(SendArgs),
|
||||||
|
Workspace(WorkspaceArgs),
|
||||||
|
Request(RequestArgs),
|
||||||
|
Folder(FolderArgs),
|
||||||
|
Environment(EnvironmentArgs),
|
||||||
|
}
|
||||||
|
```
|
||||||
|
3. Move existing `Workspaces` logic into `workspace list`
|
||||||
|
4. Move existing `Requests` logic into `request list`
|
||||||
|
5. Move existing `Send` logic into `request send`
|
||||||
|
6. Move existing `Create` logic into `request create`
|
||||||
|
7. Delete the `Get` command entirely
|
||||||
|
8. Extract shared setup (DB init, plugin init, encryption) into a reusable context struct
|
||||||
|
|
||||||
|
### Phase 2: Add missing CRUD commands
|
||||||
|
|
||||||
|
1. `workspace show <id>`
|
||||||
|
2. `workspace create --name <name>` (and `--json`)
|
||||||
|
3. `workspace update --json`
|
||||||
|
4. `workspace delete <id>`
|
||||||
|
5. `request show <id>` (JSON output of the full request model)
|
||||||
|
6. `request delete <id>`
|
||||||
|
7. `folder list <workspace_id>`
|
||||||
|
8. `folder show <id>`
|
||||||
|
9. `folder create <workspace_id> --name <name>` (and `--json`)
|
||||||
|
10. `folder update --json`
|
||||||
|
11. `folder delete <id>`
|
||||||
|
12. `environment list <workspace_id>`
|
||||||
|
13. `environment show <id>`
|
||||||
|
14. `environment create <workspace_id> --name <name>` (and `--json`)
|
||||||
|
15. `environment update --json`
|
||||||
|
16. `environment delete <id>`
|
||||||
|
|
||||||
|
### Phase 3: JSON input for create/update
|
||||||
|
|
||||||
|
Both commands accept JSON via `--json <string>` or as a positional argument (detected
|
||||||
|
by leading `{`). They follow the same upsert pattern as the plugin API.
|
||||||
|
|
||||||
|
- **`create --json`**: JSON must include `workspaceId`. Must NOT include `id` (or
|
||||||
|
use empty string `""`). Deserializes into the model with defaults for missing fields,
|
||||||
|
then upserts (insert).
|
||||||
|
- **`update --json`**: JSON must include `id`. Performs a fetch-merge-upsert:
|
||||||
|
1. Fetch the existing model from DB
|
||||||
|
2. Serialize it to `serde_json::Value`
|
||||||
|
3. Deep-merge the user's partial JSON on top (JSON Merge Patch / RFC 7386 semantics)
|
||||||
|
4. Deserialize back into the typed model
|
||||||
|
5. Upsert (update)
|
||||||
|
|
||||||
|
This matches how the MCP server plugin already does it (fetch existing, spread, override),
|
||||||
|
but the CLI handles the merge server-side so callers don't have to.
|
||||||
|
|
||||||
|
Setting a field to `null` removes it (for `Option<T>` fields), per RFC 7386.
|
||||||
|
|
||||||
|
Implementation:
|
||||||
|
1. Add `--json` flag and positional JSON detection to `create` commands
|
||||||
|
2. Add `update` commands with required `--json` flag
|
||||||
|
3. Implement JSON merge utility (or use `json-patch` crate)
|
||||||
|
|
||||||
|
### Phase 4: Runtime schema generation
|
||||||
|
|
||||||
|
1. Add `schemars` dependency to `yaak-models`
|
||||||
|
2. Derive `JsonSchema` on `HttpRequest`, `GrpcRequest`, `WebsocketRequest`, and their
|
||||||
|
nested types (`HttpRequestHeader`, `HttpUrlParameter`, etc.)
|
||||||
|
3. Implement `request schema` command:
|
||||||
|
- Generate base schema from schemars
|
||||||
|
- Query plugins for auth strategy form inputs
|
||||||
|
- Convert plugin form inputs into JSON Schema properties
|
||||||
|
- Merge into the `authentication` field
|
||||||
|
- Print to stdout
|
||||||
|
|
||||||
|
### Phase 5: Polymorphic send
|
||||||
|
|
||||||
|
1. Update `request send` to use `get_any_request` to resolve the request type
|
||||||
|
2. Match on `AnyRequest` variant and dispatch to the appropriate sender:
|
||||||
|
- `AnyRequest::HttpRequest` — existing HTTP send logic
|
||||||
|
- `AnyRequest::GrpcRequest` — gRPC invoke (future implementation)
|
||||||
|
- `AnyRequest::WebsocketRequest` — WebSocket connect (future implementation)
|
||||||
|
3. gRPC and WebSocket send can initially return "not yet implemented" errors
|
||||||
|
|
||||||
|
### Phase 6: Top-level `send` and folder/workspace send
|
||||||
|
|
||||||
|
1. Add top-level `yaakcli send <id>` command
|
||||||
|
2. Resolve ID by trying DB lookups in order: any_request → folder → workspace
|
||||||
|
3. For folder: list all requests in folder, send each
|
||||||
|
4. For workspace: list all requests in workspace, send each
|
||||||
|
5. Add execution options: `--sequential` (default), `--parallel`, `--fail-fast`
|
||||||
|
|
||||||
|
## Crate Changes
|
||||||
|
|
||||||
|
- **yaak-cli**: restructure into modules, new clap hierarchy
|
||||||
|
- **yaak-models**: add `schemars` dependency, derive `JsonSchema` on model structs
|
||||||
|
(current derives: `Debug, Clone, PartialEq, Serialize, Deserialize, Default, TS`)
|
||||||
@@ -2,7 +2,6 @@ use crate::PluginContextExt;
|
|||||||
use crate::error::Result;
|
use crate::error::Result;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use tauri::{AppHandle, Manager, Runtime, State, WebviewWindow, command};
|
use tauri::{AppHandle, Manager, Runtime, State, WebviewWindow, command};
|
||||||
use tauri_plugin_dialog::{DialogExt, MessageDialogKind};
|
|
||||||
use yaak_crypto::manager::EncryptionManager;
|
use yaak_crypto::manager::EncryptionManager;
|
||||||
use yaak_models::models::HttpRequestHeader;
|
use yaak_models::models::HttpRequestHeader;
|
||||||
use yaak_models::queries::workspaces::default_headers;
|
use yaak_models::queries::workspaces::default_headers;
|
||||||
@@ -23,20 +22,6 @@ impl<'a, R: Runtime, M: Manager<R>> EncryptionManagerExt<'a, R> for M {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_show_workspace_key<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
workspace_id: &str,
|
|
||||||
) -> Result<()> {
|
|
||||||
let key = window.crypto().reveal_workspace_key(workspace_id)?;
|
|
||||||
window
|
|
||||||
.dialog()
|
|
||||||
.message(format!("Your workspace key is \n\n{}", key))
|
|
||||||
.kind(MessageDialogKind::Info)
|
|
||||||
.show(|_v| {});
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
#[command]
|
||||||
pub(crate) async fn cmd_decrypt_template<R: Runtime>(
|
pub(crate) async fn cmd_decrypt_template<R: Runtime>(
|
||||||
window: WebviewWindow<R>,
|
window: WebviewWindow<R>,
|
||||||
|
|||||||
@@ -9,8 +9,8 @@ use yaak_git::{
|
|||||||
BranchDeleteResult, CloneResult, GitCommit, GitRemote, GitStatusSummary, PullResult,
|
BranchDeleteResult, CloneResult, GitCommit, GitRemote, GitStatusSummary, PullResult,
|
||||||
PushResult, git_add, git_add_credential, git_add_remote, git_checkout_branch, git_clone,
|
PushResult, git_add, git_add_credential, git_add_remote, git_checkout_branch, git_clone,
|
||||||
git_commit, git_create_branch, git_delete_branch, git_delete_remote_branch, git_fetch_all,
|
git_commit, git_create_branch, git_delete_branch, git_delete_remote_branch, git_fetch_all,
|
||||||
git_init, git_log, git_merge_branch, git_pull, git_push, git_remotes, git_rename_branch,
|
git_init, git_log, git_merge_branch, git_pull, git_pull_force_reset, git_pull_merge, git_push,
|
||||||
git_rm_remote, git_status, git_unstage,
|
git_remotes, git_rename_branch, git_reset_changes, git_rm_remote, git_status, git_unstage,
|
||||||
};
|
};
|
||||||
|
|
||||||
// NOTE: All of these commands are async to prevent blocking work from locking up the UI
|
// NOTE: All of these commands are async to prevent blocking work from locking up the UI
|
||||||
@@ -89,6 +89,20 @@ pub async fn cmd_git_pull(dir: &Path) -> Result<PullResult> {
|
|||||||
Ok(git_pull(dir).await?)
|
Ok(git_pull(dir).await?)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[command]
|
||||||
|
pub async fn cmd_git_pull_force_reset(
|
||||||
|
dir: &Path,
|
||||||
|
remote: &str,
|
||||||
|
branch: &str,
|
||||||
|
) -> Result<PullResult> {
|
||||||
|
Ok(git_pull_force_reset(dir, remote, branch).await?)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[command]
|
||||||
|
pub async fn cmd_git_pull_merge(dir: &Path, remote: &str, branch: &str) -> Result<PullResult> {
|
||||||
|
Ok(git_pull_merge(dir, remote, branch).await?)
|
||||||
|
}
|
||||||
|
|
||||||
#[command]
|
#[command]
|
||||||
pub async fn cmd_git_add(dir: &Path, rela_paths: Vec<PathBuf>) -> Result<()> {
|
pub async fn cmd_git_add(dir: &Path, rela_paths: Vec<PathBuf>) -> Result<()> {
|
||||||
for path in rela_paths {
|
for path in rela_paths {
|
||||||
@@ -105,6 +119,11 @@ pub async fn cmd_git_unstage(dir: &Path, rela_paths: Vec<PathBuf>) -> Result<()>
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[command]
|
||||||
|
pub async fn cmd_git_reset_changes(dir: &Path) -> Result<()> {
|
||||||
|
Ok(git_reset_changes(dir).await?)
|
||||||
|
}
|
||||||
|
|
||||||
#[command]
|
#[command]
|
||||||
pub async fn cmd_git_add_credential(
|
pub async fn cmd_git_add_credential(
|
||||||
remote_url: &str,
|
remote_url: &str,
|
||||||
|
|||||||
@@ -182,7 +182,14 @@ async fn send_http_request_inner<R: Runtime>(
|
|||||||
);
|
);
|
||||||
let env_chain =
|
let env_chain =
|
||||||
window.db().resolve_environments(&workspace.id, folder_id, environment_id.as_deref())?;
|
window.db().resolve_environments(&workspace.id, folder_id, environment_id.as_deref())?;
|
||||||
let request = render_http_request(&resolved, env_chain, &cb, &RenderOptions::throw()).await?;
|
let mut cancel_rx = cancelled_rx.clone();
|
||||||
|
let render_options = RenderOptions::throw();
|
||||||
|
let request = tokio::select! {
|
||||||
|
result = render_http_request(&resolved, env_chain, &cb, &render_options) => result?,
|
||||||
|
_ = cancel_rx.changed() => {
|
||||||
|
return Err(GenericError("Request canceled".to_string()));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// Build the sendable request using the new SendableHttpRequest type
|
// Build the sendable request using the new SendableHttpRequest type
|
||||||
let options = SendableHttpRequestOptions {
|
let options = SendableHttpRequestOptions {
|
||||||
@@ -244,16 +251,22 @@ async fn send_http_request_inner<R: Runtime>(
|
|||||||
})
|
})
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
// Apply authentication to the request
|
// Apply authentication to the request, racing against cancellation since
|
||||||
apply_authentication(
|
// auth plugins (e.g. OAuth2) can block indefinitely waiting for user action.
|
||||||
&window,
|
let mut cancel_rx = cancelled_rx.clone();
|
||||||
&mut sendable_request,
|
tokio::select! {
|
||||||
&request,
|
result = apply_authentication(
|
||||||
auth_context_id,
|
&window,
|
||||||
&plugin_manager,
|
&mut sendable_request,
|
||||||
plugin_context,
|
&request,
|
||||||
)
|
auth_context_id,
|
||||||
.await?;
|
&plugin_manager,
|
||||||
|
plugin_context,
|
||||||
|
) => result?,
|
||||||
|
_ = cancel_rx.changed() => {
|
||||||
|
return Err(GenericError("Request canceled".to_string()));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let cookie_store = maybe_cookie_store.as_ref().map(|(cs, _)| cs.clone());
|
let cookie_store = maybe_cookie_store.as_ref().map(|(cs, _)| cs.clone());
|
||||||
let result = execute_transaction(
|
let result = execute_transaction(
|
||||||
|
|||||||
@@ -37,8 +37,8 @@ use yaak_grpc::{Code, ServiceDefinition, serialize_message};
|
|||||||
use yaak_mac_window::AppHandleMacWindowExt;
|
use yaak_mac_window::AppHandleMacWindowExt;
|
||||||
use yaak_models::models::{
|
use yaak_models::models::{
|
||||||
AnyModel, CookieJar, Environment, GrpcConnection, GrpcConnectionState, GrpcEvent,
|
AnyModel, CookieJar, Environment, GrpcConnection, GrpcConnectionState, GrpcEvent,
|
||||||
GrpcEventType, GrpcRequest, HttpRequest, HttpResponse, HttpResponseEvent, HttpResponseState,
|
GrpcEventType, HttpRequest, HttpResponse, HttpResponseEvent, HttpResponseState, Plugin,
|
||||||
Plugin, Workspace, WorkspaceMeta,
|
Workspace, WorkspaceMeta,
|
||||||
};
|
};
|
||||||
use yaak_models::util::{BatchUpsertResult, UpdateSource, get_workspace_export_resources};
|
use yaak_models::util::{BatchUpsertResult, UpdateSource, get_workspace_export_resources};
|
||||||
use yaak_plugins::events::{
|
use yaak_plugins::events::{
|
||||||
@@ -1271,35 +1271,6 @@ async fn cmd_save_response<R: Runtime>(
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
async fn cmd_send_folder<R: Runtime>(
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
environment_id: Option<String>,
|
|
||||||
cookie_jar_id: Option<String>,
|
|
||||||
folder_id: &str,
|
|
||||||
) -> YaakResult<()> {
|
|
||||||
let requests = app_handle.db().list_http_requests_for_folder_recursive(folder_id)?;
|
|
||||||
for request in requests {
|
|
||||||
let app_handle = app_handle.clone();
|
|
||||||
let window = window.clone();
|
|
||||||
let environment_id = environment_id.clone();
|
|
||||||
let cookie_jar_id = cookie_jar_id.clone();
|
|
||||||
tokio::spawn(async move {
|
|
||||||
let _ = cmd_send_http_request(
|
|
||||||
app_handle,
|
|
||||||
window,
|
|
||||||
environment_id.as_deref(),
|
|
||||||
cookie_jar_id.as_deref(),
|
|
||||||
request,
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
async fn cmd_send_http_request<R: Runtime>(
|
async fn cmd_send_http_request<R: Runtime>(
|
||||||
app_handle: AppHandle<R>,
|
app_handle: AppHandle<R>,
|
||||||
@@ -1396,27 +1367,6 @@ async fn cmd_install_plugin<R: Runtime>(
|
|||||||
Ok(plugin)
|
Ok(plugin)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
async fn cmd_create_grpc_request<R: Runtime>(
|
|
||||||
workspace_id: &str,
|
|
||||||
name: &str,
|
|
||||||
sort_priority: f64,
|
|
||||||
folder_id: Option<&str>,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> YaakResult<GrpcRequest> {
|
|
||||||
Ok(app_handle.db().upsert_grpc_request(
|
|
||||||
&GrpcRequest {
|
|
||||||
workspace_id: workspace_id.to_string(),
|
|
||||||
name: name.to_string(),
|
|
||||||
folder_id: folder_id.map(|s| s.to_string()),
|
|
||||||
sort_priority,
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
async fn cmd_reload_plugins<R: Runtime>(
|
async fn cmd_reload_plugins<R: Runtime>(
|
||||||
app_handle: AppHandle<R>,
|
app_handle: AppHandle<R>,
|
||||||
@@ -1679,7 +1629,6 @@ pub fn run() {
|
|||||||
cmd_call_folder_action,
|
cmd_call_folder_action,
|
||||||
cmd_call_grpc_request_action,
|
cmd_call_grpc_request_action,
|
||||||
cmd_check_for_updates,
|
cmd_check_for_updates,
|
||||||
cmd_create_grpc_request,
|
|
||||||
cmd_curl_to_request,
|
cmd_curl_to_request,
|
||||||
cmd_delete_all_grpc_connections,
|
cmd_delete_all_grpc_connections,
|
||||||
cmd_delete_all_http_responses,
|
cmd_delete_all_http_responses,
|
||||||
@@ -1713,7 +1662,6 @@ pub fn run() {
|
|||||||
cmd_save_response,
|
cmd_save_response,
|
||||||
cmd_send_ephemeral_request,
|
cmd_send_ephemeral_request,
|
||||||
cmd_send_http_request,
|
cmd_send_http_request,
|
||||||
cmd_send_folder,
|
|
||||||
cmd_template_function_config,
|
cmd_template_function_config,
|
||||||
cmd_template_function_summaries,
|
cmd_template_function_summaries,
|
||||||
cmd_template_tokens_to_string,
|
cmd_template_tokens_to_string,
|
||||||
@@ -1728,7 +1676,6 @@ pub fn run() {
|
|||||||
crate::commands::cmd_reveal_workspace_key,
|
crate::commands::cmd_reveal_workspace_key,
|
||||||
crate::commands::cmd_secure_template,
|
crate::commands::cmd_secure_template,
|
||||||
crate::commands::cmd_set_workspace_key,
|
crate::commands::cmd_set_workspace_key,
|
||||||
crate::commands::cmd_show_workspace_key,
|
|
||||||
//
|
//
|
||||||
// Models commands
|
// Models commands
|
||||||
models_ext::models_delete,
|
models_ext::models_delete,
|
||||||
@@ -1762,8 +1709,11 @@ pub fn run() {
|
|||||||
git_ext::cmd_git_fetch_all,
|
git_ext::cmd_git_fetch_all,
|
||||||
git_ext::cmd_git_push,
|
git_ext::cmd_git_push,
|
||||||
git_ext::cmd_git_pull,
|
git_ext::cmd_git_pull,
|
||||||
|
git_ext::cmd_git_pull_force_reset,
|
||||||
|
git_ext::cmd_git_pull_merge,
|
||||||
git_ext::cmd_git_add,
|
git_ext::cmd_git_add,
|
||||||
git_ext::cmd_git_unstage,
|
git_ext::cmd_git_unstage,
|
||||||
|
git_ext::cmd_git_reset_changes,
|
||||||
git_ext::cmd_git_add_credential,
|
git_ext::cmd_git_add_credential,
|
||||||
git_ext::cmd_git_remotes,
|
git_ext::cmd_git_remotes,
|
||||||
git_ext::cmd_git_add_remote,
|
git_ext::cmd_git_add_remote,
|
||||||
@@ -1777,14 +1727,7 @@ pub fn run() {
|
|||||||
plugins_ext::cmd_plugins_update_all,
|
plugins_ext::cmd_plugins_update_all,
|
||||||
//
|
//
|
||||||
// WebSocket commands
|
// WebSocket commands
|
||||||
ws_ext::cmd_ws_upsert_request,
|
|
||||||
ws_ext::cmd_ws_duplicate_request,
|
|
||||||
ws_ext::cmd_ws_delete_request,
|
|
||||||
ws_ext::cmd_ws_delete_connection,
|
|
||||||
ws_ext::cmd_ws_delete_connections,
|
ws_ext::cmd_ws_delete_connections,
|
||||||
ws_ext::cmd_ws_list_events,
|
|
||||||
ws_ext::cmd_ws_list_requests,
|
|
||||||
ws_ext::cmd_ws_list_connections,
|
|
||||||
ws_ext::cmd_ws_send,
|
ws_ext::cmd_ws_send,
|
||||||
ws_ext::cmd_ws_close,
|
ws_ext::cmd_ws_close,
|
||||||
ws_ext::cmd_ws_connect,
|
ws_ext::cmd_ws_connect,
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ use chrono::Utc;
|
|||||||
use cookie::Cookie;
|
use cookie::Cookie;
|
||||||
use log::error;
|
use log::error;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use tauri::{AppHandle, Emitter, Manager, Runtime};
|
use tauri::{AppHandle, Emitter, Listener, Manager, Runtime};
|
||||||
use tauri_plugin_clipboard_manager::ClipboardExt;
|
use tauri_plugin_clipboard_manager::ClipboardExt;
|
||||||
use tauri_plugin_opener::OpenerExt;
|
use tauri_plugin_opener::OpenerExt;
|
||||||
use yaak_crypto::manager::EncryptionManager;
|
use yaak_crypto::manager::EncryptionManager;
|
||||||
@@ -59,7 +59,55 @@ pub(crate) async fn handle_plugin_event<R: Runtime>(
|
|||||||
}
|
}
|
||||||
InternalEventPayload::PromptFormRequest(_) => {
|
InternalEventPayload::PromptFormRequest(_) => {
|
||||||
let window = get_window_from_plugin_context(app_handle, &plugin_context)?;
|
let window = get_window_from_plugin_context(app_handle, &plugin_context)?;
|
||||||
Ok(call_frontend(&window, event).await)
|
if event.reply_id.is_some() {
|
||||||
|
// Follow-up update from plugin runtime with resolved inputs — forward to frontend
|
||||||
|
window.emit_to(window.label(), "plugin_event", event.clone())?;
|
||||||
|
Ok(None)
|
||||||
|
} else {
|
||||||
|
// Initial request — set up bidirectional communication
|
||||||
|
window.emit_to(window.label(), "plugin_event", event.clone()).unwrap();
|
||||||
|
|
||||||
|
let event_id = event.id.clone();
|
||||||
|
let plugin_handle = plugin_handle.clone();
|
||||||
|
let plugin_context = plugin_context.clone();
|
||||||
|
let window = window.clone();
|
||||||
|
|
||||||
|
// Spawn async task to handle bidirectional form communication
|
||||||
|
tauri::async_runtime::spawn(async move {
|
||||||
|
let (tx, mut rx) = tokio::sync::mpsc::channel::<InternalEvent>(128);
|
||||||
|
|
||||||
|
// Listen for replies from the frontend
|
||||||
|
let listener_id = window.listen(event_id, move |ev: tauri::Event| {
|
||||||
|
let resp: InternalEvent = serde_json::from_str(ev.payload()).unwrap();
|
||||||
|
let _ = tx.try_send(resp);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Forward each reply to the plugin runtime
|
||||||
|
while let Some(resp) = rx.recv().await {
|
||||||
|
let is_done = matches!(
|
||||||
|
&resp.payload,
|
||||||
|
InternalEventPayload::PromptFormResponse(r) if r.done.unwrap_or(false)
|
||||||
|
);
|
||||||
|
|
||||||
|
let event_to_send = plugin_handle.build_event_to_send(
|
||||||
|
&plugin_context,
|
||||||
|
&resp.payload,
|
||||||
|
Some(resp.reply_id.unwrap_or_default()),
|
||||||
|
);
|
||||||
|
if let Err(e) = plugin_handle.send(&event_to_send).await {
|
||||||
|
log::warn!("Failed to forward form response to plugin: {:?}", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
if is_done {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
window.unlisten(listener_id);
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
InternalEventPayload::FindHttpResponsesRequest(req) => {
|
InternalEventPayload::FindHttpResponsesRequest(req) => {
|
||||||
let http_responses = app_handle
|
let http_responses = app_handle
|
||||||
|
|||||||
@@ -162,11 +162,16 @@ pub(crate) fn create_window<R: Runtime>(
|
|||||||
"dev.reset_size" => webview_window
|
"dev.reset_size" => webview_window
|
||||||
.set_size(LogicalSize::new(DEFAULT_WINDOW_WIDTH, DEFAULT_WINDOW_HEIGHT))
|
.set_size(LogicalSize::new(DEFAULT_WINDOW_WIDTH, DEFAULT_WINDOW_HEIGHT))
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
"dev.reset_size_record" => {
|
"dev.reset_size_16x9" => {
|
||||||
let width = webview_window.outer_size().unwrap().width;
|
let width = webview_window.outer_size().unwrap().width;
|
||||||
let height = width * 9 / 16;
|
let height = width * 9 / 16;
|
||||||
webview_window.set_size(PhysicalSize::new(width, height)).unwrap()
|
webview_window.set_size(PhysicalSize::new(width, height)).unwrap()
|
||||||
}
|
}
|
||||||
|
"dev.reset_size_16x10" => {
|
||||||
|
let width = webview_window.outer_size().unwrap().width;
|
||||||
|
let height = width * 10 / 16;
|
||||||
|
webview_window.set_size(PhysicalSize::new(width, height)).unwrap()
|
||||||
|
}
|
||||||
"dev.refresh" => webview_window.eval("location.reload()").unwrap(),
|
"dev.refresh" => webview_window.eval("location.reload()").unwrap(),
|
||||||
"dev.generate_theme_css" => {
|
"dev.generate_theme_css" => {
|
||||||
w.emit("generate_theme_css", true).unwrap();
|
w.emit("generate_theme_css", true).unwrap();
|
||||||
|
|||||||
@@ -154,8 +154,13 @@ pub fn app_menu<R: Runtime>(app_handle: &AppHandle<R>) -> tauri::Result<Menu<R>>
|
|||||||
&MenuItemBuilder::with_id("dev.reset_size".to_string(), "Reset Size")
|
&MenuItemBuilder::with_id("dev.reset_size".to_string(), "Reset Size")
|
||||||
.build(app_handle)?,
|
.build(app_handle)?,
|
||||||
&MenuItemBuilder::with_id(
|
&MenuItemBuilder::with_id(
|
||||||
"dev.reset_size_record".to_string(),
|
"dev.reset_size_16x9".to_string(),
|
||||||
"Reset Size 16x9",
|
"Resize to 16x9",
|
||||||
|
)
|
||||||
|
.build(app_handle)?,
|
||||||
|
&MenuItemBuilder::with_id(
|
||||||
|
"dev.reset_size_16x10".to_string(),
|
||||||
|
"Resize to 16x10",
|
||||||
)
|
)
|
||||||
.build(app_handle)?,
|
.build(app_handle)?,
|
||||||
&MenuItemBuilder::with_id(
|
&MenuItemBuilder::with_id(
|
||||||
|
|||||||
@@ -28,52 +28,6 @@ use yaak_templates::{RenderErrorBehavior, RenderOptions};
|
|||||||
use yaak_tls::find_client_certificate;
|
use yaak_tls::find_client_certificate;
|
||||||
use yaak_ws::{WebsocketManager, render_websocket_request};
|
use yaak_ws::{WebsocketManager, render_websocket_request};
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_upsert_request<R: Runtime>(
|
|
||||||
request: WebsocketRequest,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<WebsocketRequest> {
|
|
||||||
Ok(app_handle
|
|
||||||
.db()
|
|
||||||
.upsert_websocket_request(&request, &UpdateSource::from_window_label(window.label()))?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_duplicate_request<R: Runtime>(
|
|
||||||
request_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<WebsocketRequest> {
|
|
||||||
let db = app_handle.db();
|
|
||||||
let request = db.get_websocket_request(request_id)?;
|
|
||||||
Ok(db.duplicate_websocket_request(&request, &UpdateSource::from_window_label(window.label()))?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_delete_request<R: Runtime>(
|
|
||||||
request_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<WebsocketRequest> {
|
|
||||||
Ok(app_handle.db().delete_websocket_request_by_id(
|
|
||||||
request_id,
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_delete_connection<R: Runtime>(
|
|
||||||
connection_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<WebsocketConnection> {
|
|
||||||
Ok(app_handle.db().delete_websocket_connection_by_id(
|
|
||||||
connection_id,
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
#[command]
|
||||||
pub async fn cmd_ws_delete_connections<R: Runtime>(
|
pub async fn cmd_ws_delete_connections<R: Runtime>(
|
||||||
request_id: &str,
|
request_id: &str,
|
||||||
@@ -86,30 +40,6 @@ pub async fn cmd_ws_delete_connections<R: Runtime>(
|
|||||||
)?)
|
)?)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_list_events<R: Runtime>(
|
|
||||||
connection_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
) -> Result<Vec<WebsocketEvent>> {
|
|
||||||
Ok(app_handle.db().list_websocket_events(connection_id)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_list_requests<R: Runtime>(
|
|
||||||
workspace_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
) -> Result<Vec<WebsocketRequest>> {
|
|
||||||
Ok(app_handle.db().list_websocket_requests(workspace_id)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_list_connections<R: Runtime>(
|
|
||||||
workspace_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
) -> Result<Vec<WebsocketConnection>> {
|
|
||||||
Ok(app_handle.db().list_websocket_connections(workspace_id)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
#[command]
|
||||||
pub async fn cmd_ws_send<R: Runtime>(
|
pub async fn cmd_ws_send<R: Runtime>(
|
||||||
connection_id: &str,
|
connection_id: &str,
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ publish = false
|
|||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
chrono = { workspace = true, features = ["serde"] }
|
chrono = { workspace = true, features = ["serde"] }
|
||||||
git2 = { version = "0.20.0", features = ["vendored-libgit2", "vendored-openssl"] }
|
git2 = { version = "0.20.4", features = ["vendored-libgit2", "vendored-openssl"] }
|
||||||
log = { workspace = true }
|
log = { workspace = true }
|
||||||
serde = { workspace = true, features = ["derive"] }
|
serde = { workspace = true, features = ["derive"] }
|
||||||
serde_json = { workspace = true }
|
serde_json = { workspace = true }
|
||||||
|
|||||||
4
crates/yaak-git/bindings/gen_git.ts
generated
4
crates/yaak-git/bindings/gen_git.ts
generated
@@ -15,8 +15,8 @@ export type GitStatus = "untracked" | "conflict" | "current" | "modified" | "rem
|
|||||||
|
|
||||||
export type GitStatusEntry = { relaPath: string, status: GitStatus, staged: boolean, prev: SyncModel | null, next: SyncModel | null, };
|
export type GitStatusEntry = { relaPath: string, status: GitStatus, staged: boolean, prev: SyncModel | null, next: SyncModel | null, };
|
||||||
|
|
||||||
export type GitStatusSummary = { path: string, headRef: string | null, headRefShorthand: string | null, entries: Array<GitStatusEntry>, origins: Array<string>, localBranches: Array<string>, remoteBranches: Array<string>, };
|
export type GitStatusSummary = { path: string, headRef: string | null, headRefShorthand: string | null, entries: Array<GitStatusEntry>, origins: Array<string>, localBranches: Array<string>, remoteBranches: Array<string>, ahead: number, behind: number, };
|
||||||
|
|
||||||
export type PullResult = { "type": "success", message: string, } | { "type": "up_to_date" } | { "type": "needs_credentials", url: string, error: string | null, };
|
export type PullResult = { "type": "success", message: string, } | { "type": "up_to_date" } | { "type": "needs_credentials", url: string, error: string | null, } | { "type": "diverged", remote: string, branch: string, } | { "type": "uncommitted_changes" };
|
||||||
|
|
||||||
export type PushResult = { "type": "success", message: string, } | { "type": "up_to_date" } | { "type": "needs_credentials", url: string, error: string | null, };
|
export type PushResult = { "type": "success", message: string, } | { "type": "up_to_date" } | { "type": "needs_credentials", url: string, error: string | null, };
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { createFastMutation } from '@yaakapp/app/hooks/useFastMutation';
|
|||||||
import { queryClient } from '@yaakapp/app/lib/queryClient';
|
import { queryClient } from '@yaakapp/app/lib/queryClient';
|
||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { BranchDeleteResult, CloneResult, GitCommit, GitRemote, GitStatusSummary, PullResult, PushResult } from './bindings/gen_git';
|
import { BranchDeleteResult, CloneResult, GitCommit, GitRemote, GitStatusSummary, PullResult, PushResult } from './bindings/gen_git';
|
||||||
|
import { showToast } from '@yaakapp/app/lib/toast';
|
||||||
|
|
||||||
export * from './bindings/gen_git';
|
export * from './bindings/gen_git';
|
||||||
export * from './bindings/gen_models';
|
export * from './bindings/gen_models';
|
||||||
@@ -13,11 +14,20 @@ export interface GitCredentials {
|
|||||||
password: string;
|
password: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type DivergedStrategy = 'force_reset' | 'merge' | 'cancel';
|
||||||
|
|
||||||
|
export type UncommittedChangesStrategy = 'reset' | 'cancel';
|
||||||
|
|
||||||
export interface GitCallbacks {
|
export interface GitCallbacks {
|
||||||
addRemote: () => Promise<GitRemote | null>;
|
addRemote: () => Promise<GitRemote | null>;
|
||||||
promptCredentials: (
|
promptCredentials: (
|
||||||
result: Extract<PushResult, { type: 'needs_credentials' }>,
|
result: Extract<PushResult, { type: 'needs_credentials' }>,
|
||||||
) => Promise<GitCredentials | null>;
|
) => Promise<GitCredentials | null>;
|
||||||
|
promptDiverged: (
|
||||||
|
result: Extract<PullResult, { type: 'diverged' }>,
|
||||||
|
) => Promise<DivergedStrategy>;
|
||||||
|
promptUncommittedChanges: () => Promise<UncommittedChangesStrategy>;
|
||||||
|
forceSync: () => Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
const onSuccess = () => queryClient.invalidateQueries({ queryKey: ['git'] });
|
const onSuccess = () => queryClient.invalidateQueries({ queryKey: ['git'] });
|
||||||
@@ -69,6 +79,15 @@ export const gitMutations = (dir: string, callbacks: GitCallbacks) => {
|
|||||||
return invoke<PushResult>('cmd_git_push', { dir });
|
return invoke<PushResult>('cmd_git_push', { dir });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleError = (err: unknown) => {
|
||||||
|
showToast({
|
||||||
|
id: `${err}`,
|
||||||
|
message: `${err}`,
|
||||||
|
color: 'danger',
|
||||||
|
timeout: 5000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
init: createFastMutation<void, string, void>({
|
init: createFastMutation<void, string, void>({
|
||||||
mutationKey: ['git', 'init'],
|
mutationKey: ['git', 'init'],
|
||||||
@@ -133,10 +152,9 @@ export const gitMutations = (dir: string, callbacks: GitCallbacks) => {
|
|||||||
},
|
},
|
||||||
onSuccess,
|
onSuccess,
|
||||||
}),
|
}),
|
||||||
fetchAll: createFastMutation<string, string, void>({
|
fetchAll: createFastMutation<void, string, void>({
|
||||||
mutationKey: ['git', 'checkout', dir],
|
mutationKey: ['git', 'fetch_all', dir],
|
||||||
mutationFn: () => invoke('cmd_git_fetch_all', { dir }),
|
mutationFn: () => invoke('cmd_git_fetch_all', { dir }),
|
||||||
onSuccess,
|
|
||||||
}),
|
}),
|
||||||
push: createFastMutation<PushResult, string, void>({
|
push: createFastMutation<PushResult, string, void>({
|
||||||
mutationKey: ['git', 'push', dir],
|
mutationKey: ['git', 'push', dir],
|
||||||
@@ -147,20 +165,51 @@ export const gitMutations = (dir: string, callbacks: GitCallbacks) => {
|
|||||||
mutationKey: ['git', 'pull', dir],
|
mutationKey: ['git', 'pull', dir],
|
||||||
async mutationFn() {
|
async mutationFn() {
|
||||||
const result = await invoke<PullResult>('cmd_git_pull', { dir });
|
const result = await invoke<PullResult>('cmd_git_pull', { dir });
|
||||||
if (result.type !== 'needs_credentials') return result;
|
|
||||||
|
|
||||||
// Needs credentials, prompt for them
|
if (result.type === 'needs_credentials') {
|
||||||
const creds = await callbacks.promptCredentials(result);
|
const creds = await callbacks.promptCredentials(result);
|
||||||
if (creds == null) throw new Error('Canceled');
|
if (creds == null) throw new Error('Canceled');
|
||||||
|
|
||||||
await invoke('cmd_git_add_credential', {
|
await invoke('cmd_git_add_credential', {
|
||||||
remoteUrl: result.url,
|
remoteUrl: result.url,
|
||||||
username: creds.username,
|
username: creds.username,
|
||||||
password: creds.password,
|
password: creds.password,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Pull again
|
// Pull again after credentials
|
||||||
return invoke<PullResult>('cmd_git_pull', { dir });
|
return invoke<PullResult>('cmd_git_pull', { dir });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.type === 'uncommitted_changes') {
|
||||||
|
callbacks.promptUncommittedChanges().then(async (strategy) => {
|
||||||
|
if (strategy === 'cancel') return;
|
||||||
|
|
||||||
|
await invoke('cmd_git_reset_changes', { dir });
|
||||||
|
return invoke<PullResult>('cmd_git_pull', { dir });
|
||||||
|
}).then(async () => { onSuccess(); await callbacks.forceSync(); }, handleError);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.type === 'diverged') {
|
||||||
|
callbacks.promptDiverged(result).then((strategy) => {
|
||||||
|
if (strategy === 'cancel') return;
|
||||||
|
|
||||||
|
if (strategy === 'force_reset') {
|
||||||
|
return invoke<PullResult>('cmd_git_pull_force_reset', {
|
||||||
|
dir,
|
||||||
|
remote: result.remote,
|
||||||
|
branch: result.branch,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return invoke<PullResult>('cmd_git_pull_merge', {
|
||||||
|
dir,
|
||||||
|
remote: result.remote,
|
||||||
|
branch: result.branch,
|
||||||
|
});
|
||||||
|
}).then(async () => { onSuccess(); await callbacks.forceSync(); }, handleError);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
},
|
},
|
||||||
onSuccess,
|
onSuccess,
|
||||||
}),
|
}),
|
||||||
@@ -169,6 +218,11 @@ export const gitMutations = (dir: string, callbacks: GitCallbacks) => {
|
|||||||
mutationFn: (args) => invoke('cmd_git_unstage', { dir, ...args }),
|
mutationFn: (args) => invoke('cmd_git_unstage', { dir, ...args }),
|
||||||
onSuccess,
|
onSuccess,
|
||||||
}),
|
}),
|
||||||
|
resetChanges: createFastMutation<void, string, void>({
|
||||||
|
mutationKey: ['git', 'reset-changes', dir],
|
||||||
|
mutationFn: () => invoke('cmd_git_reset_changes', { dir }),
|
||||||
|
onSuccess,
|
||||||
|
}),
|
||||||
} as const;
|
} as const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ mod pull;
|
|||||||
mod push;
|
mod push;
|
||||||
mod remotes;
|
mod remotes;
|
||||||
mod repository;
|
mod repository;
|
||||||
|
mod reset;
|
||||||
mod status;
|
mod status;
|
||||||
mod unstage;
|
mod unstage;
|
||||||
mod util;
|
mod util;
|
||||||
@@ -29,8 +30,9 @@ pub use credential::git_add_credential;
|
|||||||
pub use fetch::git_fetch_all;
|
pub use fetch::git_fetch_all;
|
||||||
pub use init::git_init;
|
pub use init::git_init;
|
||||||
pub use log::{GitCommit, git_log};
|
pub use log::{GitCommit, git_log};
|
||||||
pub use pull::{PullResult, git_pull};
|
pub use pull::{PullResult, git_pull, git_pull_force_reset, git_pull_merge};
|
||||||
pub use push::{PushResult, git_push};
|
pub use push::{PushResult, git_push};
|
||||||
pub use remotes::{GitRemote, git_add_remote, git_remotes, git_rm_remote};
|
pub use remotes::{GitRemote, git_add_remote, git_remotes, git_rm_remote};
|
||||||
|
pub use reset::git_reset_changes;
|
||||||
pub use status::{GitStatusSummary, git_status};
|
pub use status::{GitStatusSummary, git_status};
|
||||||
pub use unstage::git_unstage;
|
pub use unstage::git_unstage;
|
||||||
|
|||||||
@@ -15,9 +15,23 @@ pub enum PullResult {
|
|||||||
Success { message: String },
|
Success { message: String },
|
||||||
UpToDate,
|
UpToDate,
|
||||||
NeedsCredentials { url: String, error: Option<String> },
|
NeedsCredentials { url: String, error: Option<String> },
|
||||||
|
Diverged { remote: String, branch: String },
|
||||||
|
UncommittedChanges,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn has_uncommitted_changes(dir: &Path) -> Result<bool> {
|
||||||
|
let repo = open_repo(dir)?;
|
||||||
|
let mut opts = git2::StatusOptions::new();
|
||||||
|
opts.include_ignored(false).include_untracked(false);
|
||||||
|
let statuses = repo.statuses(Some(&mut opts))?;
|
||||||
|
Ok(statuses.iter().any(|e| e.status() != git2::Status::CURRENT))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn git_pull(dir: &Path) -> Result<PullResult> {
|
pub async fn git_pull(dir: &Path) -> Result<PullResult> {
|
||||||
|
if has_uncommitted_changes(dir)? {
|
||||||
|
return Ok(PullResult::UncommittedChanges);
|
||||||
|
}
|
||||||
|
|
||||||
// Extract all git2 data before any await points (git2 types are not Send)
|
// Extract all git2 data before any await points (git2 types are not Send)
|
||||||
let (branch_name, remote_name, remote_url) = {
|
let (branch_name, remote_name, remote_url) = {
|
||||||
let repo = open_repo(dir)?;
|
let repo = open_repo(dir)?;
|
||||||
@@ -56,6 +70,13 @@ pub async fn git_pull(dir: &Path) -> Result<PullResult> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if !out.status.success() {
|
if !out.status.success() {
|
||||||
|
let combined_lower = combined.to_lowercase();
|
||||||
|
if combined_lower.contains("cannot fast-forward")
|
||||||
|
|| combined_lower.contains("not possible to fast-forward")
|
||||||
|
|| combined_lower.contains("diverged")
|
||||||
|
{
|
||||||
|
return Ok(PullResult::Diverged { remote: remote_name, branch: branch_name });
|
||||||
|
}
|
||||||
return Err(GenericError(format!("Failed to pull {combined}")));
|
return Err(GenericError(format!("Failed to pull {combined}")));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -66,6 +87,65 @@ pub async fn git_pull(dir: &Path) -> Result<PullResult> {
|
|||||||
Ok(PullResult::Success { message: format!("Pulled from {}/{}", remote_name, branch_name) })
|
Ok(PullResult::Success { message: format!("Pulled from {}/{}", remote_name, branch_name) })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn git_pull_force_reset(dir: &Path, remote: &str, branch: &str) -> Result<PullResult> {
|
||||||
|
// Step 1: fetch the remote
|
||||||
|
let fetch_out = new_binary_command(dir)
|
||||||
|
.await?
|
||||||
|
.args(["fetch", remote])
|
||||||
|
.env("GIT_TERMINAL_PROMPT", "0")
|
||||||
|
.output()
|
||||||
|
.await
|
||||||
|
.map_err(|e| GenericError(format!("failed to run git fetch: {e}")))?;
|
||||||
|
|
||||||
|
if !fetch_out.status.success() {
|
||||||
|
let stderr = String::from_utf8_lossy(&fetch_out.stderr);
|
||||||
|
return Err(GenericError(format!("Failed to fetch: {stderr}")));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 2: reset --hard to remote/branch
|
||||||
|
let ref_name = format!("{}/{}", remote, branch);
|
||||||
|
let reset_out = new_binary_command(dir)
|
||||||
|
.await?
|
||||||
|
.args(["reset", "--hard", &ref_name])
|
||||||
|
.output()
|
||||||
|
.await
|
||||||
|
.map_err(|e| GenericError(format!("failed to run git reset: {e}")))?;
|
||||||
|
|
||||||
|
if !reset_out.status.success() {
|
||||||
|
let stderr = String::from_utf8_lossy(&reset_out.stderr);
|
||||||
|
return Err(GenericError(format!("Failed to reset: {}", stderr.trim())));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(PullResult::Success { message: format!("Reset to {}/{}", remote, branch) })
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn git_pull_merge(dir: &Path, remote: &str, branch: &str) -> Result<PullResult> {
|
||||||
|
let out = new_binary_command(dir)
|
||||||
|
.await?
|
||||||
|
.args(["pull", "--no-rebase", remote, branch])
|
||||||
|
.env("GIT_TERMINAL_PROMPT", "0")
|
||||||
|
.output()
|
||||||
|
.await
|
||||||
|
.map_err(|e| GenericError(format!("failed to run git pull --no-rebase: {e}")))?;
|
||||||
|
|
||||||
|
let stdout = String::from_utf8_lossy(&out.stdout);
|
||||||
|
let stderr = String::from_utf8_lossy(&out.stderr);
|
||||||
|
let combined = format!("{}{}", stdout, stderr);
|
||||||
|
|
||||||
|
info!("Pull merge status={} {combined}", out.status);
|
||||||
|
|
||||||
|
if !out.status.success() {
|
||||||
|
if combined.to_lowercase().contains("conflict") {
|
||||||
|
return Err(GenericError(
|
||||||
|
"Merge conflicts detected. Please resolve them manually.".to_string(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
return Err(GenericError(format!("Failed to merge pull: {}", combined.trim())));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(PullResult::Success { message: format!("Merged from {}/{}", remote, branch) })
|
||||||
|
}
|
||||||
|
|
||||||
// pub(crate) fn git_pull_old(dir: &Path) -> Result<PullResult> {
|
// pub(crate) fn git_pull_old(dir: &Path) -> Result<PullResult> {
|
||||||
// let repo = open_repo(dir)?;
|
// let repo = open_repo(dir)?;
|
||||||
//
|
//
|
||||||
|
|||||||
20
crates/yaak-git/src/reset.rs
Normal file
20
crates/yaak-git/src/reset.rs
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
use crate::binary::new_binary_command;
|
||||||
|
use crate::error::Error::GenericError;
|
||||||
|
use crate::error::Result;
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
pub async fn git_reset_changes(dir: &Path) -> Result<()> {
|
||||||
|
let out = new_binary_command(dir)
|
||||||
|
.await?
|
||||||
|
.args(["reset", "--hard", "HEAD"])
|
||||||
|
.output()
|
||||||
|
.await
|
||||||
|
.map_err(|e| GenericError(format!("failed to run git reset: {e}")))?;
|
||||||
|
|
||||||
|
if !out.status.success() {
|
||||||
|
let stderr = String::from_utf8_lossy(&out.stderr);
|
||||||
|
return Err(GenericError(format!("Failed to reset: {}", stderr.trim())));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
@@ -18,6 +18,8 @@ pub struct GitStatusSummary {
|
|||||||
pub origins: Vec<String>,
|
pub origins: Vec<String>,
|
||||||
pub local_branches: Vec<String>,
|
pub local_branches: Vec<String>,
|
||||||
pub remote_branches: Vec<String>,
|
pub remote_branches: Vec<String>,
|
||||||
|
pub ahead: u32,
|
||||||
|
pub behind: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize, TS)]
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize, TS)]
|
||||||
@@ -160,6 +162,18 @@ pub fn git_status(dir: &Path) -> crate::error::Result<GitStatusSummary> {
|
|||||||
let local_branches = local_branch_names(&repo)?;
|
let local_branches = local_branch_names(&repo)?;
|
||||||
let remote_branches = remote_branch_names(&repo)?;
|
let remote_branches = remote_branch_names(&repo)?;
|
||||||
|
|
||||||
|
// Compute ahead/behind relative to remote tracking branch
|
||||||
|
let (ahead, behind) = (|| -> Option<(usize, usize)> {
|
||||||
|
let head = repo.head().ok()?;
|
||||||
|
let local_oid = head.target()?;
|
||||||
|
let branch_name = head.shorthand()?;
|
||||||
|
let upstream_ref =
|
||||||
|
repo.find_branch(&format!("origin/{branch_name}"), git2::BranchType::Remote).ok()?;
|
||||||
|
let upstream_oid = upstream_ref.get().target()?;
|
||||||
|
repo.graph_ahead_behind(local_oid, upstream_oid).ok()
|
||||||
|
})()
|
||||||
|
.unwrap_or((0, 0));
|
||||||
|
|
||||||
Ok(GitStatusSummary {
|
Ok(GitStatusSummary {
|
||||||
entries,
|
entries,
|
||||||
origins,
|
origins,
|
||||||
@@ -168,5 +182,7 @@ pub fn git_status(dir: &Path) -> crate::error::Result<GitStatusSummary> {
|
|||||||
head_ref_shorthand,
|
head_ref_shorthand,
|
||||||
local_branches,
|
local_branches,
|
||||||
remote_branches,
|
remote_branches,
|
||||||
|
ahead: ahead as u32,
|
||||||
|
behind: behind as u32,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ publish = false
|
|||||||
async-compression = { version = "0.4", features = ["tokio", "gzip", "deflate", "brotli", "zstd"] }
|
async-compression = { version = "0.4", features = ["tokio", "gzip", "deflate", "brotli", "zstd"] }
|
||||||
async-trait = "0.1"
|
async-trait = "0.1"
|
||||||
brotli = "7"
|
brotli = "7"
|
||||||
bytes = "1.5.0"
|
bytes = "1.11.1"
|
||||||
cookie = "0.18.1"
|
cookie = "0.18.1"
|
||||||
flate2 = "1"
|
flate2 = "1"
|
||||||
futures-util = "0.3"
|
futures-util = "0.3"
|
||||||
|
|||||||
@@ -168,6 +168,7 @@ impl<S: HttpSender> HttpTransaction<S> {
|
|||||||
response.drain().await?;
|
response.drain().await?;
|
||||||
|
|
||||||
// Update the request URL
|
// Update the request URL
|
||||||
|
let previous_url = current_url.clone();
|
||||||
current_url = if location.starts_with("http://") || location.starts_with("https://") {
|
current_url = if location.starts_with("http://") || location.starts_with("https://") {
|
||||||
// Absolute URL
|
// Absolute URL
|
||||||
location
|
location
|
||||||
@@ -181,6 +182,8 @@ impl<S: HttpSender> HttpTransaction<S> {
|
|||||||
format!("{}/{}", base_path, location)
|
format!("{}/{}", base_path, location)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Self::remove_sensitive_headers(&mut current_headers, &previous_url, ¤t_url);
|
||||||
|
|
||||||
// Determine redirect behavior based on status code and method
|
// Determine redirect behavior based on status code and method
|
||||||
let behavior = if status == 303 {
|
let behavior = if status == 303 {
|
||||||
// 303 See Other always changes to GET
|
// 303 See Other always changes to GET
|
||||||
@@ -220,6 +223,33 @@ impl<S: HttpSender> HttpTransaction<S> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Remove sensitive headers when redirecting to a different host.
|
||||||
|
/// This matches reqwest's `remove_sensitive_headers()` behavior and prevents
|
||||||
|
/// credentials from being forwarded to third-party servers (e.g., an
|
||||||
|
/// Authorization header sent from an API redirect to an S3 bucket).
|
||||||
|
fn remove_sensitive_headers(
|
||||||
|
headers: &mut Vec<(String, String)>,
|
||||||
|
previous_url: &str,
|
||||||
|
next_url: &str,
|
||||||
|
) {
|
||||||
|
let previous_host = Url::parse(previous_url).ok().and_then(|u| {
|
||||||
|
u.host_str().map(|h| format!("{}:{}", h, u.port_or_known_default().unwrap_or(0)))
|
||||||
|
});
|
||||||
|
let next_host = Url::parse(next_url).ok().and_then(|u| {
|
||||||
|
u.host_str().map(|h| format!("{}:{}", h, u.port_or_known_default().unwrap_or(0)))
|
||||||
|
});
|
||||||
|
if previous_host != next_host {
|
||||||
|
headers.retain(|h| {
|
||||||
|
let name_lower = h.0.to_lowercase();
|
||||||
|
name_lower != "authorization"
|
||||||
|
&& name_lower != "cookie"
|
||||||
|
&& name_lower != "cookie2"
|
||||||
|
&& name_lower != "proxy-authorization"
|
||||||
|
&& name_lower != "www-authenticate"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Check if a status code indicates a redirect
|
/// Check if a status code indicates a redirect
|
||||||
fn is_redirect(status: u16) -> bool {
|
fn is_redirect(status: u16) -> bool {
|
||||||
matches!(status, 301 | 302 | 303 | 307 | 308)
|
matches!(status, 301 | 302 | 303 | 307 | 308)
|
||||||
@@ -269,9 +299,20 @@ mod tests {
|
|||||||
use tokio::io::AsyncRead;
|
use tokio::io::AsyncRead;
|
||||||
use tokio::sync::Mutex;
|
use tokio::sync::Mutex;
|
||||||
|
|
||||||
|
/// Captured request metadata for test assertions
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
#[allow(dead_code)]
|
||||||
|
struct CapturedRequest {
|
||||||
|
url: String,
|
||||||
|
method: String,
|
||||||
|
headers: Vec<(String, String)>,
|
||||||
|
}
|
||||||
|
|
||||||
/// Mock sender for testing
|
/// Mock sender for testing
|
||||||
struct MockSender {
|
struct MockSender {
|
||||||
responses: Arc<Mutex<Vec<MockResponse>>>,
|
responses: Arc<Mutex<Vec<MockResponse>>>,
|
||||||
|
/// Captured requests for assertions
|
||||||
|
captured_requests: Arc<Mutex<Vec<CapturedRequest>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct MockResponse {
|
struct MockResponse {
|
||||||
@@ -282,7 +323,10 @@ mod tests {
|
|||||||
|
|
||||||
impl MockSender {
|
impl MockSender {
|
||||||
fn new(responses: Vec<MockResponse>) -> Self {
|
fn new(responses: Vec<MockResponse>) -> Self {
|
||||||
Self { responses: Arc::new(Mutex::new(responses)) }
|
Self {
|
||||||
|
responses: Arc::new(Mutex::new(responses)),
|
||||||
|
captured_requests: Arc::new(Mutex::new(Vec::new())),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -290,9 +334,16 @@ mod tests {
|
|||||||
impl HttpSender for MockSender {
|
impl HttpSender for MockSender {
|
||||||
async fn send(
|
async fn send(
|
||||||
&self,
|
&self,
|
||||||
_request: SendableHttpRequest,
|
request: SendableHttpRequest,
|
||||||
_event_tx: mpsc::Sender<HttpResponseEvent>,
|
_event_tx: mpsc::Sender<HttpResponseEvent>,
|
||||||
) -> Result<HttpResponse> {
|
) -> Result<HttpResponse> {
|
||||||
|
// Capture the request metadata for later assertions
|
||||||
|
self.captured_requests.lock().await.push(CapturedRequest {
|
||||||
|
url: request.url.clone(),
|
||||||
|
method: request.method.clone(),
|
||||||
|
headers: request.headers.clone(),
|
||||||
|
});
|
||||||
|
|
||||||
let mut responses = self.responses.lock().await;
|
let mut responses = self.responses.lock().await;
|
||||||
if responses.is_empty() {
|
if responses.is_empty() {
|
||||||
Err(crate::error::Error::RequestError("No more mock responses".to_string()))
|
Err(crate::error::Error::RequestError("No more mock responses".to_string()))
|
||||||
@@ -726,4 +777,116 @@ mod tests {
|
|||||||
assert!(result.is_ok());
|
assert!(result.is_ok());
|
||||||
assert_eq!(request_count.load(Ordering::SeqCst), 2);
|
assert_eq!(request_count.load(Ordering::SeqCst), 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_cross_origin_redirect_strips_auth_headers() {
|
||||||
|
// Redirect from api.example.com -> s3.amazonaws.com should strip Authorization
|
||||||
|
let responses = vec![
|
||||||
|
MockResponse {
|
||||||
|
status: 302,
|
||||||
|
headers: vec![(
|
||||||
|
"Location".to_string(),
|
||||||
|
"https://s3.amazonaws.com/bucket/file.pdf".to_string(),
|
||||||
|
)],
|
||||||
|
body: vec![],
|
||||||
|
},
|
||||||
|
MockResponse { status: 200, headers: Vec::new(), body: b"PDF content".to_vec() },
|
||||||
|
];
|
||||||
|
|
||||||
|
let sender = MockSender::new(responses);
|
||||||
|
let captured = sender.captured_requests.clone();
|
||||||
|
let transaction = HttpTransaction::new(sender);
|
||||||
|
|
||||||
|
let request = SendableHttpRequest {
|
||||||
|
url: "https://api.example.com/download".to_string(),
|
||||||
|
method: "GET".to_string(),
|
||||||
|
headers: vec![
|
||||||
|
("Authorization".to_string(), "Basic dXNlcjpwYXNz".to_string()),
|
||||||
|
("Accept".to_string(), "application/pdf".to_string()),
|
||||||
|
],
|
||||||
|
options: crate::types::SendableHttpRequestOptions {
|
||||||
|
follow_redirects: true,
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
|
||||||
|
let (_tx, rx) = tokio::sync::watch::channel(false);
|
||||||
|
let (event_tx, _event_rx) = mpsc::channel(100);
|
||||||
|
let result = transaction.execute_with_cancellation(request, rx, event_tx).await.unwrap();
|
||||||
|
assert_eq!(result.status, 200);
|
||||||
|
|
||||||
|
let requests = captured.lock().await;
|
||||||
|
assert_eq!(requests.len(), 2);
|
||||||
|
|
||||||
|
// First request should have the Authorization header
|
||||||
|
assert!(
|
||||||
|
requests[0].headers.iter().any(|(k, _)| k.eq_ignore_ascii_case("authorization")),
|
||||||
|
"First request should have Authorization header"
|
||||||
|
);
|
||||||
|
|
||||||
|
// Second request (to different host) should NOT have the Authorization header
|
||||||
|
assert!(
|
||||||
|
!requests[1].headers.iter().any(|(k, _)| k.eq_ignore_ascii_case("authorization")),
|
||||||
|
"Redirected request to different host should NOT have Authorization header"
|
||||||
|
);
|
||||||
|
|
||||||
|
// Non-sensitive headers should still be present
|
||||||
|
assert!(
|
||||||
|
requests[1].headers.iter().any(|(k, _)| k.eq_ignore_ascii_case("accept")),
|
||||||
|
"Non-sensitive headers should be preserved across cross-origin redirects"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_same_origin_redirect_preserves_auth_headers() {
|
||||||
|
// Redirect within the same host should keep Authorization
|
||||||
|
let responses = vec![
|
||||||
|
MockResponse {
|
||||||
|
status: 302,
|
||||||
|
headers: vec![(
|
||||||
|
"Location".to_string(),
|
||||||
|
"https://api.example.com/v2/download".to_string(),
|
||||||
|
)],
|
||||||
|
body: vec![],
|
||||||
|
},
|
||||||
|
MockResponse { status: 200, headers: Vec::new(), body: b"OK".to_vec() },
|
||||||
|
];
|
||||||
|
|
||||||
|
let sender = MockSender::new(responses);
|
||||||
|
let captured = sender.captured_requests.clone();
|
||||||
|
let transaction = HttpTransaction::new(sender);
|
||||||
|
|
||||||
|
let request = SendableHttpRequest {
|
||||||
|
url: "https://api.example.com/v1/download".to_string(),
|
||||||
|
method: "GET".to_string(),
|
||||||
|
headers: vec![
|
||||||
|
("Authorization".to_string(), "Bearer token123".to_string()),
|
||||||
|
("Accept".to_string(), "application/json".to_string()),
|
||||||
|
],
|
||||||
|
options: crate::types::SendableHttpRequestOptions {
|
||||||
|
follow_redirects: true,
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
|
||||||
|
let (_tx, rx) = tokio::sync::watch::channel(false);
|
||||||
|
let (event_tx, _event_rx) = mpsc::channel(100);
|
||||||
|
let result = transaction.execute_with_cancellation(request, rx, event_tx).await.unwrap();
|
||||||
|
assert_eq!(result.status, 200);
|
||||||
|
|
||||||
|
let requests = captured.lock().await;
|
||||||
|
assert_eq!(requests.len(), 2);
|
||||||
|
|
||||||
|
// Both requests should have the Authorization header (same host)
|
||||||
|
assert!(
|
||||||
|
requests[0].headers.iter().any(|(k, _)| k.eq_ignore_ascii_case("authorization")),
|
||||||
|
"First request should have Authorization header"
|
||||||
|
);
|
||||||
|
assert!(
|
||||||
|
requests[1].headers.iter().any(|(k, _)| k.eq_ignore_ascii_case("authorization")),
|
||||||
|
"Redirected request to same host should preserve Authorization header"
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
14
crates/yaak-plugins/bindings/gen_events.ts
generated
14
crates/yaak-plugins/bindings/gen_events.ts
generated
@@ -66,7 +66,9 @@ export type DeleteModelRequest = { model: string, id: string, };
|
|||||||
|
|
||||||
export type DeleteModelResponse = { model: AnyModel, };
|
export type DeleteModelResponse = { model: AnyModel, };
|
||||||
|
|
||||||
export type EditorLanguage = "text" | "javascript" | "json" | "html" | "xml" | "graphql" | "markdown";
|
export type DialogSize = "sm" | "md" | "lg" | "full" | "dynamic";
|
||||||
|
|
||||||
|
export type EditorLanguage = "text" | "javascript" | "json" | "html" | "xml" | "graphql" | "markdown" | "c" | "clojure" | "csharp" | "go" | "http" | "java" | "kotlin" | "objective_c" | "ocaml" | "php" | "powershell" | "python" | "r" | "ruby" | "shell" | "swift";
|
||||||
|
|
||||||
export type EmptyPayload = {};
|
export type EmptyPayload = {};
|
||||||
|
|
||||||
@@ -172,7 +174,11 @@ hideGutter?: boolean,
|
|||||||
/**
|
/**
|
||||||
* Language for syntax highlighting
|
* Language for syntax highlighting
|
||||||
*/
|
*/
|
||||||
language?: EditorLanguage, readOnly?: boolean, completionOptions?: Array<GenericCompletionOption>,
|
language?: EditorLanguage, readOnly?: boolean,
|
||||||
|
/**
|
||||||
|
* Fixed number of visible rows
|
||||||
|
*/
|
||||||
|
rows?: number, completionOptions?: Array<GenericCompletionOption>,
|
||||||
/**
|
/**
|
||||||
* The name of the input. The value will be stored at this object attribute in the resulting data
|
* The name of the input. The value will be stored at this object attribute in the resulting data
|
||||||
*/
|
*/
|
||||||
@@ -476,9 +482,9 @@ label: string, title?: string, size?: WindowSize, dataDirKey?: string, };
|
|||||||
|
|
||||||
export type PluginContext = { id: string, label: string | null, workspaceId: string | null, };
|
export type PluginContext = { id: string, label: string | null, workspaceId: string | null, };
|
||||||
|
|
||||||
export type PromptFormRequest = { id: string, title: string, description?: string, inputs: Array<FormInput>, confirmText?: string, cancelText?: string, };
|
export type PromptFormRequest = { id: string, title: string, description?: string, inputs: Array<FormInput>, confirmText?: string, cancelText?: string, size?: DialogSize, };
|
||||||
|
|
||||||
export type PromptFormResponse = { values: { [key in string]?: JsonPrimitive } | null, };
|
export type PromptFormResponse = { values: { [key in string]?: JsonPrimitive } | null, done?: boolean, };
|
||||||
|
|
||||||
export type PromptTextRequest = { id: string, title: string, label: string, description?: string, defaultValue?: string, placeholder?: string,
|
export type PromptTextRequest = { id: string, title: string, label: string, description?: string, defaultValue?: string, placeholder?: string,
|
||||||
/**
|
/**
|
||||||
|
|||||||
2
crates/yaak-plugins/bindings/gen_models.ts
generated
2
crates/yaak-plugins/bindings/gen_models.ts
generated
@@ -49,7 +49,7 @@ export type HttpResponseEvent = { model: "http_response_event", id: string, crea
|
|||||||
* This mirrors `yaak_http::sender::HttpResponseEvent` but with serde support.
|
* This mirrors `yaak_http::sender::HttpResponseEvent` but with serde support.
|
||||||
* The `From` impl is in yaak-http to avoid circular dependencies.
|
* The `From` impl is in yaak-http to avoid circular dependencies.
|
||||||
*/
|
*/
|
||||||
export type HttpResponseEventData = { "type": "setting", name: string, value: string, } | { "type": "info", message: string, } | { "type": "redirect", url: string, status: number, behavior: string, } | { "type": "send_url", method: string, path: string, } | { "type": "receive_url", version: string, status: string, } | { "type": "header_up", name: string, value: string, } | { "type": "header_down", name: string, value: string, } | { "type": "chunk_sent", bytes: number, } | { "type": "chunk_received", bytes: number, } | { "type": "dns_resolved", hostname: string, addresses: Array<string>, duration: bigint, overridden: boolean, };
|
export type HttpResponseEventData = { "type": "setting", name: string, value: string, } | { "type": "info", message: string, } | { "type": "redirect", url: string, status: number, behavior: string, } | { "type": "send_url", method: string, scheme: string, username: string, password: string, host: string, port: number, path: string, query: string, fragment: string, } | { "type": "receive_url", version: string, status: string, } | { "type": "header_up", name: string, value: string, } | { "type": "header_down", name: string, value: string, } | { "type": "chunk_sent", bytes: number, } | { "type": "chunk_received", bytes: number, } | { "type": "dns_resolved", hostname: string, addresses: Array<string>, duration: bigint, overridden: boolean, };
|
||||||
|
|
||||||
export type HttpResponseHeader = { name: string, value: string, };
|
export type HttpResponseHeader = { name: string, value: string, };
|
||||||
|
|
||||||
|
|||||||
@@ -587,6 +587,19 @@ pub struct PromptFormRequest {
|
|||||||
pub confirm_text: Option<String>,
|
pub confirm_text: Option<String>,
|
||||||
#[ts(optional)]
|
#[ts(optional)]
|
||||||
pub cancel_text: Option<String>,
|
pub cancel_text: Option<String>,
|
||||||
|
#[ts(optional)]
|
||||||
|
pub size: Option<DialogSize>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Serialize, Deserialize, TS)]
|
||||||
|
#[serde(rename_all = "snake_case")]
|
||||||
|
#[ts(export, export_to = "gen_events.ts")]
|
||||||
|
pub enum DialogSize {
|
||||||
|
Sm,
|
||||||
|
Md,
|
||||||
|
Lg,
|
||||||
|
Full,
|
||||||
|
Dynamic,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, Serialize, Deserialize, TS)]
|
#[derive(Debug, Clone, Default, Serialize, Deserialize, TS)]
|
||||||
@@ -594,6 +607,8 @@ pub struct PromptFormRequest {
|
|||||||
#[ts(export, export_to = "gen_events.ts")]
|
#[ts(export, export_to = "gen_events.ts")]
|
||||||
pub struct PromptFormResponse {
|
pub struct PromptFormResponse {
|
||||||
pub values: Option<HashMap<String, JsonPrimitive>>,
|
pub values: Option<HashMap<String, JsonPrimitive>>,
|
||||||
|
#[ts(optional)]
|
||||||
|
pub done: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, Serialize, Deserialize, TS)]
|
#[derive(Debug, Clone, Default, Serialize, Deserialize, TS)]
|
||||||
@@ -936,6 +951,22 @@ pub enum EditorLanguage {
|
|||||||
Xml,
|
Xml,
|
||||||
Graphql,
|
Graphql,
|
||||||
Markdown,
|
Markdown,
|
||||||
|
C,
|
||||||
|
Clojure,
|
||||||
|
Csharp,
|
||||||
|
Go,
|
||||||
|
Http,
|
||||||
|
Java,
|
||||||
|
Kotlin,
|
||||||
|
ObjectiveC,
|
||||||
|
Ocaml,
|
||||||
|
Php,
|
||||||
|
Powershell,
|
||||||
|
Python,
|
||||||
|
R,
|
||||||
|
Ruby,
|
||||||
|
Shell,
|
||||||
|
Swift,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for EditorLanguage {
|
impl Default for EditorLanguage {
|
||||||
@@ -966,6 +997,10 @@ pub struct FormInputEditor {
|
|||||||
#[ts(optional)]
|
#[ts(optional)]
|
||||||
pub read_only: Option<bool>,
|
pub read_only: Option<bool>,
|
||||||
|
|
||||||
|
/// Fixed number of visible rows
|
||||||
|
#[ts(optional)]
|
||||||
|
pub rows: Option<i32>,
|
||||||
|
|
||||||
#[ts(optional)]
|
#[ts(optional)]
|
||||||
pub completion_options: Option<Vec<GenericCompletionOption>>,
|
pub completion_options: Option<Vec<GenericCompletionOption>>,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ use std::time::Duration;
|
|||||||
use tokio::fs::read_dir;
|
use tokio::fs::read_dir;
|
||||||
use tokio::net::TcpListener;
|
use tokio::net::TcpListener;
|
||||||
use tokio::sync::mpsc::error::TrySendError;
|
use tokio::sync::mpsc::error::TrySendError;
|
||||||
use tokio::sync::{Mutex, mpsc};
|
use tokio::sync::{Mutex, mpsc, oneshot};
|
||||||
use tokio::time::{Instant, timeout};
|
use tokio::time::{Instant, timeout};
|
||||||
use yaak_models::models::Plugin;
|
use yaak_models::models::Plugin;
|
||||||
use yaak_models::util::generate_id;
|
use yaak_models::util::generate_id;
|
||||||
@@ -43,6 +43,7 @@ pub struct PluginManager {
|
|||||||
subscribers: Arc<Mutex<HashMap<String, mpsc::Sender<InternalEvent>>>>,
|
subscribers: Arc<Mutex<HashMap<String, mpsc::Sender<InternalEvent>>>>,
|
||||||
plugin_handles: Arc<Mutex<Vec<PluginHandle>>>,
|
plugin_handles: Arc<Mutex<Vec<PluginHandle>>>,
|
||||||
kill_tx: tokio::sync::watch::Sender<bool>,
|
kill_tx: tokio::sync::watch::Sender<bool>,
|
||||||
|
killed_rx: Arc<Mutex<Option<oneshot::Receiver<()>>>>,
|
||||||
ws_service: Arc<PluginRuntimeServerWebsocket>,
|
ws_service: Arc<PluginRuntimeServerWebsocket>,
|
||||||
vendored_plugin_dir: PathBuf,
|
vendored_plugin_dir: PathBuf,
|
||||||
pub(crate) installed_plugin_dir: PathBuf,
|
pub(crate) installed_plugin_dir: PathBuf,
|
||||||
@@ -70,6 +71,7 @@ impl PluginManager {
|
|||||||
) -> PluginManager {
|
) -> PluginManager {
|
||||||
let (events_tx, mut events_rx) = mpsc::channel(2048);
|
let (events_tx, mut events_rx) = mpsc::channel(2048);
|
||||||
let (kill_server_tx, kill_server_rx) = tokio::sync::watch::channel(false);
|
let (kill_server_tx, kill_server_rx) = tokio::sync::watch::channel(false);
|
||||||
|
let (killed_tx, killed_rx) = oneshot::channel();
|
||||||
|
|
||||||
let (client_disconnect_tx, mut client_disconnect_rx) = mpsc::channel(128);
|
let (client_disconnect_tx, mut client_disconnect_rx) = mpsc::channel(128);
|
||||||
let (client_connect_tx, mut client_connect_rx) = tokio::sync::watch::channel(false);
|
let (client_connect_tx, mut client_connect_rx) = tokio::sync::watch::channel(false);
|
||||||
@@ -81,6 +83,7 @@ impl PluginManager {
|
|||||||
subscribers: Default::default(),
|
subscribers: Default::default(),
|
||||||
ws_service: Arc::new(ws_service.clone()),
|
ws_service: Arc::new(ws_service.clone()),
|
||||||
kill_tx: kill_server_tx,
|
kill_tx: kill_server_tx,
|
||||||
|
killed_rx: Arc::new(Mutex::new(Some(killed_rx))),
|
||||||
vendored_plugin_dir,
|
vendored_plugin_dir,
|
||||||
installed_plugin_dir,
|
installed_plugin_dir,
|
||||||
dev_mode,
|
dev_mode,
|
||||||
@@ -141,9 +144,15 @@ impl PluginManager {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// 2. Start Node.js runtime
|
// 2. Start Node.js runtime
|
||||||
start_nodejs_plugin_runtime(&node_bin_path, &plugin_runtime_main, addr, &kill_server_rx)
|
start_nodejs_plugin_runtime(
|
||||||
.await
|
&node_bin_path,
|
||||||
.unwrap();
|
&plugin_runtime_main,
|
||||||
|
addr,
|
||||||
|
&kill_server_rx,
|
||||||
|
killed_tx,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
info!("Waiting for plugins to initialize");
|
info!("Waiting for plugins to initialize");
|
||||||
init_plugins_task.await.unwrap();
|
init_plugins_task.await.unwrap();
|
||||||
|
|
||||||
@@ -296,8 +305,15 @@ impl PluginManager {
|
|||||||
pub async fn terminate(&self) {
|
pub async fn terminate(&self) {
|
||||||
self.kill_tx.send_replace(true);
|
self.kill_tx.send_replace(true);
|
||||||
|
|
||||||
// Give it a bit of time to kill
|
// Wait for the plugin runtime process to actually exit
|
||||||
tokio::time::sleep(Duration::from_millis(500)).await;
|
let killed_rx = self.killed_rx.lock().await.take();
|
||||||
|
if let Some(rx) = killed_rx {
|
||||||
|
if timeout(Duration::from_secs(5), rx).await.is_err() {
|
||||||
|
warn!("Timed out waiting for plugin runtime to exit");
|
||||||
|
} else {
|
||||||
|
info!("Plugin runtime exited")
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn reply(
|
pub async fn reply(
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ use std::net::SocketAddr;
|
|||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::process::Stdio;
|
use std::process::Stdio;
|
||||||
use tokio::io::{AsyncBufReadExt, BufReader};
|
use tokio::io::{AsyncBufReadExt, BufReader};
|
||||||
|
use tokio::sync::oneshot;
|
||||||
use tokio::sync::watch::Receiver;
|
use tokio::sync::watch::Receiver;
|
||||||
use yaak_common::command::new_xplatform_command;
|
use yaak_common::command::new_xplatform_command;
|
||||||
|
|
||||||
@@ -19,6 +20,7 @@ pub async fn start_nodejs_plugin_runtime(
|
|||||||
plugin_runtime_main: &Path,
|
plugin_runtime_main: &Path,
|
||||||
addr: SocketAddr,
|
addr: SocketAddr,
|
||||||
kill_rx: &Receiver<bool>,
|
kill_rx: &Receiver<bool>,
|
||||||
|
killed_tx: oneshot::Sender<()>,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
// HACK: Remove UNC prefix for Windows paths to pass to sidecar
|
// HACK: Remove UNC prefix for Windows paths to pass to sidecar
|
||||||
let plugin_runtime_main_str =
|
let plugin_runtime_main_str =
|
||||||
@@ -72,6 +74,7 @@ pub async fn start_nodejs_plugin_runtime(
|
|||||||
warn!("Failed to kill plugin runtime: {e}");
|
warn!("Failed to kill plugin runtime: {e}");
|
||||||
}
|
}
|
||||||
info!("Killed plugin runtime");
|
info!("Killed plugin runtime");
|
||||||
|
let _ = killed_tx.send(());
|
||||||
});
|
});
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|||||||
@@ -1,31 +1,5 @@
|
|||||||
import { invoke } from '@tauri-apps/api/core';
|
import { invoke } from '@tauri-apps/api/core';
|
||||||
import { WebsocketConnection, WebsocketEvent, WebsocketRequest } from '@yaakapp-internal/models';
|
import { WebsocketConnection } from '@yaakapp-internal/models';
|
||||||
|
|
||||||
export function upsertWebsocketRequest(
|
|
||||||
request: WebsocketRequest | Partial<Omit<WebsocketRequest, 'id'>>,
|
|
||||||
) {
|
|
||||||
return invoke('cmd_ws_upsert_request', {
|
|
||||||
request,
|
|
||||||
}) as Promise<WebsocketRequest>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function duplicateWebsocketRequest(requestId: string) {
|
|
||||||
return invoke('cmd_ws_duplicate_request', {
|
|
||||||
requestId,
|
|
||||||
}) as Promise<WebsocketRequest>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function deleteWebsocketRequest(requestId: string) {
|
|
||||||
return invoke('cmd_ws_delete_request', {
|
|
||||||
requestId,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function deleteWebsocketConnection(connectionId: string) {
|
|
||||||
return invoke('cmd_ws_delete_connection', {
|
|
||||||
connectionId,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function deleteWebsocketConnections(requestId: string) {
|
export function deleteWebsocketConnections(requestId: string) {
|
||||||
return invoke('cmd_ws_delete_connections', {
|
return invoke('cmd_ws_delete_connections', {
|
||||||
@@ -33,20 +7,6 @@ export function deleteWebsocketConnections(requestId: string) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function listWebsocketRequests({ workspaceId }: { workspaceId: string }) {
|
|
||||||
return invoke('cmd_ws_list_requests', { workspaceId }) as Promise<WebsocketRequest[]>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function listWebsocketEvents({ connectionId }: { connectionId: string }) {
|
|
||||||
return invoke('cmd_ws_list_events', { connectionId }) as Promise<WebsocketEvent[]>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function listWebsocketConnections({ workspaceId }: { workspaceId: string }) {
|
|
||||||
return invoke('cmd_ws_list_connections', { workspaceId }) as Promise<
|
|
||||||
WebsocketConnection[]
|
|
||||||
>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function connectWebsocket({
|
export function connectWebsocket({
|
||||||
requestId,
|
requestId,
|
||||||
environmentId,
|
environmentId,
|
||||||
|
|||||||
236
package-lock.json
generated
236
package-lock.json
generated
@@ -13,6 +13,7 @@
|
|||||||
"packages/plugin-runtime-types",
|
"packages/plugin-runtime-types",
|
||||||
"plugins-external/mcp-server",
|
"plugins-external/mcp-server",
|
||||||
"plugins-external/template-function-faker",
|
"plugins-external/template-function-faker",
|
||||||
|
"plugins-external/httpsnippet",
|
||||||
"plugins/action-copy-curl",
|
"plugins/action-copy-curl",
|
||||||
"plugins/action-copy-grpcurl",
|
"plugins/action-copy-grpcurl",
|
||||||
"plugins/action-send-folder",
|
"plugins/action-send-folder",
|
||||||
@@ -62,6 +63,13 @@
|
|||||||
"crates/yaak-ws",
|
"crates/yaak-ws",
|
||||||
"src-web"
|
"src-web"
|
||||||
],
|
],
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/lang-go": "^6.0.1",
|
||||||
|
"@codemirror/lang-java": "^6.0.2",
|
||||||
|
"@codemirror/lang-php": "^6.0.2",
|
||||||
|
"@codemirror/lang-python": "^6.2.1",
|
||||||
|
"@codemirror/legacy-modes": "^6.5.2"
|
||||||
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@biomejs/biome": "^2.3.13",
|
"@biomejs/biome": "^2.3.13",
|
||||||
"@tauri-apps/cli": "^2.9.6",
|
"@tauri-apps/cli": "^2.9.6",
|
||||||
@@ -736,6 +744,19 @@
|
|||||||
"@lezer/css": "^1.1.7"
|
"@lezer/css": "^1.1.7"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@codemirror/lang-go": {
|
||||||
|
"version": "6.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@codemirror/lang-go/-/lang-go-6.0.1.tgz",
|
||||||
|
"integrity": "sha512-7fNvbyNylvqCphW9HD6WFnRpcDjr+KXX/FgqXy5H5ZS0eC5edDljukm/yNgYkwTsgp2busdod50AOTIy6Jikfg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/autocomplete": "^6.0.0",
|
||||||
|
"@codemirror/language": "^6.6.0",
|
||||||
|
"@codemirror/state": "^6.0.0",
|
||||||
|
"@lezer/common": "^1.0.0",
|
||||||
|
"@lezer/go": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@codemirror/lang-html": {
|
"node_modules/@codemirror/lang-html": {
|
||||||
"version": "6.4.11",
|
"version": "6.4.11",
|
||||||
"resolved": "https://registry.npmjs.org/@codemirror/lang-html/-/lang-html-6.4.11.tgz",
|
"resolved": "https://registry.npmjs.org/@codemirror/lang-html/-/lang-html-6.4.11.tgz",
|
||||||
@@ -753,6 +774,16 @@
|
|||||||
"@lezer/html": "^1.3.12"
|
"@lezer/html": "^1.3.12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@codemirror/lang-java": {
|
||||||
|
"version": "6.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@codemirror/lang-java/-/lang-java-6.0.2.tgz",
|
||||||
|
"integrity": "sha512-m5Nt1mQ/cznJY7tMfQTJchmrjdjQ71IDs+55d1GAa8DGaB8JXWsVCkVT284C3RTASaY43YknrK2X3hPO/J3MOQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/language": "^6.0.0",
|
||||||
|
"@lezer/java": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@codemirror/lang-javascript": {
|
"node_modules/@codemirror/lang-javascript": {
|
||||||
"version": "6.2.4",
|
"version": "6.2.4",
|
||||||
"resolved": "https://registry.npmjs.org/@codemirror/lang-javascript/-/lang-javascript-6.2.4.tgz",
|
"resolved": "https://registry.npmjs.org/@codemirror/lang-javascript/-/lang-javascript-6.2.4.tgz",
|
||||||
@@ -793,6 +824,32 @@
|
|||||||
"@lezer/markdown": "^1.0.0"
|
"@lezer/markdown": "^1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@codemirror/lang-php": {
|
||||||
|
"version": "6.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@codemirror/lang-php/-/lang-php-6.0.2.tgz",
|
||||||
|
"integrity": "sha512-ZKy2v1n8Fc8oEXj0Th0PUMXzQJ0AIR6TaZU+PbDHExFwdu+guzOA4jmCHS1Nz4vbFezwD7LyBdDnddSJeScMCA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/lang-html": "^6.0.0",
|
||||||
|
"@codemirror/language": "^6.0.0",
|
||||||
|
"@codemirror/state": "^6.0.0",
|
||||||
|
"@lezer/common": "^1.0.0",
|
||||||
|
"@lezer/php": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@codemirror/lang-python": {
|
||||||
|
"version": "6.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@codemirror/lang-python/-/lang-python-6.2.1.tgz",
|
||||||
|
"integrity": "sha512-IRjC8RUBhn9mGR9ywecNhB51yePWCGgvHfY1lWN/Mrp3cKuHr0isDKia+9HnvhiWNnMpbGhWrkhuWOc09exRyw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/autocomplete": "^6.3.2",
|
||||||
|
"@codemirror/language": "^6.8.0",
|
||||||
|
"@codemirror/state": "^6.0.0",
|
||||||
|
"@lezer/common": "^1.2.1",
|
||||||
|
"@lezer/python": "^1.1.4"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@codemirror/lang-xml": {
|
"node_modules/@codemirror/lang-xml": {
|
||||||
"version": "6.1.0",
|
"version": "6.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/@codemirror/lang-xml/-/lang-xml-6.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/@codemirror/lang-xml/-/lang-xml-6.1.0.tgz",
|
||||||
@@ -836,6 +893,15 @@
|
|||||||
"style-mod": "^4.0.0"
|
"style-mod": "^4.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@codemirror/legacy-modes": {
|
||||||
|
"version": "6.5.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@codemirror/legacy-modes/-/legacy-modes-6.5.2.tgz",
|
||||||
|
"integrity": "sha512-/jJbwSTazlQEDOQw2FJ8LEEKVS72pU0lx6oM54kGpL8t/NJ2Jda3CZ4pcltiKTdqYSRk3ug1B3pil1gsjA6+8Q==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/language": "^6.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@codemirror/lint": {
|
"node_modules/@codemirror/lint": {
|
||||||
"version": "6.9.2",
|
"version": "6.9.2",
|
||||||
"resolved": "https://registry.npmjs.org/@codemirror/lint/-/lint-6.9.2.tgz",
|
"resolved": "https://registry.npmjs.org/@codemirror/lint/-/lint-6.9.2.tgz",
|
||||||
@@ -1414,9 +1480,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@hono/node-server": {
|
"node_modules/@hono/node-server": {
|
||||||
"version": "1.19.8",
|
"version": "1.19.9",
|
||||||
"resolved": "https://registry.npmjs.org/@hono/node-server/-/node-server-1.19.8.tgz",
|
"resolved": "https://registry.npmjs.org/@hono/node-server/-/node-server-1.19.9.tgz",
|
||||||
"integrity": "sha512-0/g2lIOPzX8f3vzW1ggQgvG5mjtFBDBHFAzI5SFAi2DzSqS9luJwqg9T6O/gKYLi+inS7eNxBeIFkkghIPvrMA==",
|
"integrity": "sha512-vHL6w3ecZsky+8P5MD+eFfaGTyCeOHUIFYMGpQGbrBTSmNNoxv0if69rEZ5giu36weC5saFuznL411gRX7bJDw==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18.14.1"
|
"node": ">=18.14.1"
|
||||||
@@ -1570,6 +1636,17 @@
|
|||||||
"lezer-generator": "src/lezer-generator.cjs"
|
"lezer-generator": "src/lezer-generator.cjs"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@lezer/go": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@lezer/go/-/go-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-xToRsYxwsgJNHTgNdStpcvmbVuKxTapV0dM0wey1geMMRc9aggoVyKgzYp41D2/vVOx+Ii4hmE206kvxIXBVXQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@lezer/common": "^1.2.0",
|
||||||
|
"@lezer/highlight": "^1.0.0",
|
||||||
|
"@lezer/lr": "^1.3.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@lezer/highlight": {
|
"node_modules/@lezer/highlight": {
|
||||||
"version": "1.2.3",
|
"version": "1.2.3",
|
||||||
"resolved": "https://registry.npmjs.org/@lezer/highlight/-/highlight-1.2.3.tgz",
|
"resolved": "https://registry.npmjs.org/@lezer/highlight/-/highlight-1.2.3.tgz",
|
||||||
@@ -1590,6 +1667,17 @@
|
|||||||
"@lezer/lr": "^1.0.0"
|
"@lezer/lr": "^1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@lezer/java": {
|
||||||
|
"version": "1.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/@lezer/java/-/java-1.1.3.tgz",
|
||||||
|
"integrity": "sha512-yHquUfujwg6Yu4Fd1GNHCvidIvJwi/1Xu2DaKl/pfWIA2c1oXkVvawH3NyXhCaFx4OdlYBVX5wvz2f7Aoa/4Xw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@lezer/common": "^1.2.0",
|
||||||
|
"@lezer/highlight": "^1.0.0",
|
||||||
|
"@lezer/lr": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@lezer/javascript": {
|
"node_modules/@lezer/javascript": {
|
||||||
"version": "1.5.4",
|
"version": "1.5.4",
|
||||||
"resolved": "https://registry.npmjs.org/@lezer/javascript/-/javascript-1.5.4.tgz",
|
"resolved": "https://registry.npmjs.org/@lezer/javascript/-/javascript-1.5.4.tgz",
|
||||||
@@ -1631,6 +1719,28 @@
|
|||||||
"@lezer/highlight": "^1.0.0"
|
"@lezer/highlight": "^1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@lezer/php": {
|
||||||
|
"version": "1.0.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/@lezer/php/-/php-1.0.5.tgz",
|
||||||
|
"integrity": "sha512-W7asp9DhM6q0W6DYNwIkLSKOvxlXRrif+UXBMxzsJUuqmhE7oVU+gS3THO4S/Puh7Xzgm858UNaFi6dxTP8dJA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@lezer/common": "^1.2.0",
|
||||||
|
"@lezer/highlight": "^1.0.0",
|
||||||
|
"@lezer/lr": "^1.1.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@lezer/python": {
|
||||||
|
"version": "1.1.18",
|
||||||
|
"resolved": "https://registry.npmjs.org/@lezer/python/-/python-1.1.18.tgz",
|
||||||
|
"integrity": "sha512-31FiUrU7z9+d/ElGQLJFXl+dKOdx0jALlP3KEOsGTex8mvj+SoE1FgItcHWK/axkxCHGUSpqIHt6JAWfWu9Rhg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@lezer/common": "^1.2.0",
|
||||||
|
"@lezer/highlight": "^1.0.0",
|
||||||
|
"@lezer/lr": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@lezer/xml": {
|
"node_modules/@lezer/xml": {
|
||||||
"version": "1.0.6",
|
"version": "1.0.6",
|
||||||
"resolved": "https://registry.npmjs.org/@lezer/xml/-/xml-1.0.6.tgz",
|
"resolved": "https://registry.npmjs.org/@lezer/xml/-/xml-1.0.6.tgz",
|
||||||
@@ -1675,12 +1785,12 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@modelcontextprotocol/sdk": {
|
"node_modules/@modelcontextprotocol/sdk": {
|
||||||
"version": "1.25.2",
|
"version": "1.26.0",
|
||||||
"resolved": "https://registry.npmjs.org/@modelcontextprotocol/sdk/-/sdk-1.25.2.tgz",
|
"resolved": "https://registry.npmjs.org/@modelcontextprotocol/sdk/-/sdk-1.26.0.tgz",
|
||||||
"integrity": "sha512-LZFeo4F9M5qOhC/Uc1aQSrBHxMrvxett+9KLHt7OhcExtoiRN9DKgbZffMP/nxjutWDQpfMDfP3nkHI4X9ijww==",
|
"integrity": "sha512-Y5RmPncpiDtTXDbLKswIJzTqu2hyBKxTNsgKqKclDbhIgg1wgtf1fRuvxgTnRfcnxtvvgbIEcqUOzZrJ6iSReg==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@hono/node-server": "^1.19.7",
|
"@hono/node-server": "^1.19.9",
|
||||||
"ajv": "^8.17.1",
|
"ajv": "^8.17.1",
|
||||||
"ajv-formats": "^3.0.1",
|
"ajv-formats": "^3.0.1",
|
||||||
"content-type": "^1.0.5",
|
"content-type": "^1.0.5",
|
||||||
@@ -1688,14 +1798,15 @@
|
|||||||
"cross-spawn": "^7.0.5",
|
"cross-spawn": "^7.0.5",
|
||||||
"eventsource": "^3.0.2",
|
"eventsource": "^3.0.2",
|
||||||
"eventsource-parser": "^3.0.0",
|
"eventsource-parser": "^3.0.0",
|
||||||
"express": "^5.0.1",
|
"express": "^5.2.1",
|
||||||
"express-rate-limit": "^7.5.0",
|
"express-rate-limit": "^8.2.1",
|
||||||
"jose": "^6.1.1",
|
"hono": "^4.11.4",
|
||||||
|
"jose": "^6.1.3",
|
||||||
"json-schema-typed": "^8.0.2",
|
"json-schema-typed": "^8.0.2",
|
||||||
"pkce-challenge": "^5.0.0",
|
"pkce-challenge": "^5.0.0",
|
||||||
"raw-body": "^3.0.0",
|
"raw-body": "^3.0.0",
|
||||||
"zod": "^3.25 || ^4.0",
|
"zod": "^3.25 || ^4.0",
|
||||||
"zod-to-json-schema": "^3.25.0"
|
"zod-to-json-schema": "^3.25.1"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18"
|
"node": ">=18"
|
||||||
@@ -2021,6 +2132,19 @@
|
|||||||
"node": ">=16.9"
|
"node": ">=16.9"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@readme/httpsnippet": {
|
||||||
|
"version": "11.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@readme/httpsnippet/-/httpsnippet-11.0.0.tgz",
|
||||||
|
"integrity": "sha512-XSyaAsJkZfmMO9R4WDlVJARZgd4wlImftSkMkKclidniXA1h6DTya9iTqJenQo9mHQLh3u6kAC3CDRaIV+LbLw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"qs": "^6.11.2",
|
||||||
|
"stringify-object": "^3.3.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@replit/codemirror-emacs": {
|
"node_modules/@replit/codemirror-emacs": {
|
||||||
"version": "6.1.0",
|
"version": "6.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/@replit/codemirror-emacs/-/codemirror-emacs-6.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/@replit/codemirror-emacs/-/codemirror-emacs-6.1.0.tgz",
|
||||||
@@ -4044,6 +4168,10 @@
|
|||||||
"resolved": "plugins/filter-xpath",
|
"resolved": "plugins/filter-xpath",
|
||||||
"link": true
|
"link": true
|
||||||
},
|
},
|
||||||
|
"node_modules/@yaak/httpsnippet": {
|
||||||
|
"resolved": "plugins-external/httpsnippet",
|
||||||
|
"link": true
|
||||||
|
},
|
||||||
"node_modules/@yaak/importer-curl": {
|
"node_modules/@yaak/importer-curl": {
|
||||||
"resolved": "plugins/importer-curl",
|
"resolved": "plugins/importer-curl",
|
||||||
"link": true
|
"link": true
|
||||||
@@ -6865,10 +6993,13 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/express-rate-limit": {
|
"node_modules/express-rate-limit": {
|
||||||
"version": "7.5.1",
|
"version": "8.2.1",
|
||||||
"resolved": "https://registry.npmjs.org/express-rate-limit/-/express-rate-limit-7.5.1.tgz",
|
"resolved": "https://registry.npmjs.org/express-rate-limit/-/express-rate-limit-8.2.1.tgz",
|
||||||
"integrity": "sha512-7iN8iPMDzOMHPUYllBEsQdWVB6fPDMPqwjBaFrgr4Jgr/+okjvzAy+UHlYYL/Vs0OsOrMkwS6PJDkFlJwoxUnw==",
|
"integrity": "sha512-PCZEIEIxqwhzw4KF0n7QF4QqruVTcF73O5kFKUnGOyjbCCgizBBiFaYpd/fnBLUMPw/BWw9OsiN7GgrNYr7j6g==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"ip-address": "10.0.1"
|
||||||
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 16"
|
"node": ">= 16"
|
||||||
},
|
},
|
||||||
@@ -7407,6 +7538,12 @@
|
|||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/get-own-enumerable-property-symbols": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/get-own-enumerable-property-symbols/-/get-own-enumerable-property-symbols-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-I0UBV/XOz1XkIJHEUDMZAbzCThU/H8DxmSfmdGcKPnVhu2VfFqr34jr9777IyaTYvxjedWhqVIilEDsCdP5G6g==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
"node_modules/get-proto": {
|
"node_modules/get-proto": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
|
||||||
@@ -8135,6 +8272,15 @@
|
|||||||
"node": ">= 0.4"
|
"node": ">= 0.4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/ip-address": {
|
||||||
|
"version": "10.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ip-address/-/ip-address-10.0.1.tgz",
|
||||||
|
"integrity": "sha512-NWv9YLW4PoW2B7xtzaS3NCot75m6nK7Icdv0o3lfMceJVRfSoQwqD4wEH5rLwoKJwUiZ/rfpiVBhnaF0FK4HoA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 12"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/ip-bigint": {
|
"node_modules/ip-bigint": {
|
||||||
"version": "7.3.0",
|
"version": "7.3.0",
|
||||||
"resolved": "https://registry.npmjs.org/ip-bigint/-/ip-bigint-7.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/ip-bigint/-/ip-bigint-7.3.0.tgz",
|
||||||
@@ -8516,6 +8662,15 @@
|
|||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/is-obj": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-l4RyHgRqGN4Y3+9JHVrNqO+tN0rV5My76uW5/nuO4K1b6vw5G8d/cmFjP9tRfEsdhZNt0IFdZuK/c2Vr4Nb+Qg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/is-plain-obj": {
|
"node_modules/is-plain-obj": {
|
||||||
"version": "4.1.0",
|
"version": "4.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-4.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-4.1.0.tgz",
|
||||||
@@ -13776,6 +13931,29 @@
|
|||||||
"url": "https://github.com/sponsors/wooorm"
|
"url": "https://github.com/sponsors/wooorm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/stringify-object": {
|
||||||
|
"version": "3.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/stringify-object/-/stringify-object-3.3.0.tgz",
|
||||||
|
"integrity": "sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw==",
|
||||||
|
"license": "BSD-2-Clause",
|
||||||
|
"dependencies": {
|
||||||
|
"get-own-enumerable-property-symbols": "^3.0.0",
|
||||||
|
"is-obj": "^1.0.1",
|
||||||
|
"is-regexp": "^1.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=4"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/stringify-object/node_modules/is-regexp": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/is-regexp/-/is-regexp-1.0.0.tgz",
|
||||||
|
"integrity": "sha512-7zjFAPO4/gwyQAAgRRmqeEeyIICSdmCqa3tsVHMdBzaXXRiqopZL4Cyghg/XulGWrtABTpbnYYzzIRffLkP4oA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/strip-ansi": {
|
"node_modules/strip-ansi": {
|
||||||
"version": "7.1.2",
|
"version": "7.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.2.tgz",
|
||||||
@@ -15775,13 +15953,41 @@
|
|||||||
"undici-types": "~7.16.0"
|
"undici-types": "~7.16.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"plugins-external/httpsnippet": {
|
||||||
|
"name": "@yaak/httpsnippet",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"dependencies": {
|
||||||
|
"@readme/httpsnippet": "^11.0.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/node": "^22.0.0",
|
||||||
|
"typescript": "^5.9.3"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"plugins-external/httpsnippet/node_modules/@types/node": {
|
||||||
|
"version": "22.19.9",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/node/-/node-22.19.9.tgz",
|
||||||
|
"integrity": "sha512-PD03/U8g1F9T9MI+1OBisaIARhSzeidsUjQaf51fOxrfjeiKN9bLVO06lHuHYjxdnqLWJijJHfqXPSJri2EM2A==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"undici-types": "~6.21.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"plugins-external/httpsnippet/node_modules/undici-types": {
|
||||||
|
"version": "6.21.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
|
||||||
|
"integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"plugins-external/mcp-server": {
|
"plugins-external/mcp-server": {
|
||||||
"name": "@yaak/mcp-server",
|
"name": "@yaak/mcp-server",
|
||||||
"version": "0.1.7",
|
"version": "0.1.7",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@hono/mcp": "^0.2.3",
|
"@hono/mcp": "^0.2.3",
|
||||||
"@hono/node-server": "^1.19.7",
|
"@hono/node-server": "^1.19.7",
|
||||||
"@modelcontextprotocol/sdk": "^1.25.2",
|
"@modelcontextprotocol/sdk": "^1.26.0",
|
||||||
"hono": "^4.11.7",
|
"hono": "^4.11.7",
|
||||||
"zod": "^3.25.76"
|
"zod": "^3.25.76"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -12,6 +12,7 @@
|
|||||||
"packages/plugin-runtime-types",
|
"packages/plugin-runtime-types",
|
||||||
"plugins-external/mcp-server",
|
"plugins-external/mcp-server",
|
||||||
"plugins-external/template-function-faker",
|
"plugins-external/template-function-faker",
|
||||||
|
"plugins-external/httpsnippet",
|
||||||
"plugins/action-copy-curl",
|
"plugins/action-copy-curl",
|
||||||
"plugins/action-copy-grpcurl",
|
"plugins/action-copy-grpcurl",
|
||||||
"plugins/action-send-folder",
|
"plugins/action-send-folder",
|
||||||
@@ -104,5 +105,12 @@
|
|||||||
"npm-run-all": "^4.1.5",
|
"npm-run-all": "^4.1.5",
|
||||||
"typescript": "^5.8.3",
|
"typescript": "^5.8.3",
|
||||||
"vitest": "^3.2.4"
|
"vitest": "^3.2.4"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@codemirror/lang-go": "^6.0.1",
|
||||||
|
"@codemirror/lang-java": "^6.0.2",
|
||||||
|
"@codemirror/lang-php": "^6.0.2",
|
||||||
|
"@codemirror/lang-python": "^6.2.1",
|
||||||
|
"@codemirror/legacy-modes": "^6.5.2"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -66,7 +66,9 @@ export type DeleteModelRequest = { model: string, id: string, };
|
|||||||
|
|
||||||
export type DeleteModelResponse = { model: AnyModel, };
|
export type DeleteModelResponse = { model: AnyModel, };
|
||||||
|
|
||||||
export type EditorLanguage = "text" | "javascript" | "json" | "html" | "xml" | "graphql" | "markdown";
|
export type DialogSize = "sm" | "md" | "lg" | "full" | "dynamic";
|
||||||
|
|
||||||
|
export type EditorLanguage = "text" | "javascript" | "json" | "html" | "xml" | "graphql" | "markdown" | "c" | "clojure" | "csharp" | "go" | "http" | "java" | "kotlin" | "objective_c" | "ocaml" | "php" | "powershell" | "python" | "r" | "ruby" | "shell" | "swift";
|
||||||
|
|
||||||
export type EmptyPayload = {};
|
export type EmptyPayload = {};
|
||||||
|
|
||||||
@@ -172,7 +174,11 @@ hideGutter?: boolean,
|
|||||||
/**
|
/**
|
||||||
* Language for syntax highlighting
|
* Language for syntax highlighting
|
||||||
*/
|
*/
|
||||||
language?: EditorLanguage, readOnly?: boolean, completionOptions?: Array<GenericCompletionOption>,
|
language?: EditorLanguage, readOnly?: boolean,
|
||||||
|
/**
|
||||||
|
* Fixed number of visible rows
|
||||||
|
*/
|
||||||
|
rows?: number, completionOptions?: Array<GenericCompletionOption>,
|
||||||
/**
|
/**
|
||||||
* The name of the input. The value will be stored at this object attribute in the resulting data
|
* The name of the input. The value will be stored at this object attribute in the resulting data
|
||||||
*/
|
*/
|
||||||
@@ -476,9 +482,9 @@ label: string, title?: string, size?: WindowSize, dataDirKey?: string, };
|
|||||||
|
|
||||||
export type PluginContext = { id: string, label: string | null, workspaceId: string | null, };
|
export type PluginContext = { id: string, label: string | null, workspaceId: string | null, };
|
||||||
|
|
||||||
export type PromptFormRequest = { id: string, title: string, description?: string, inputs: Array<FormInput>, confirmText?: string, cancelText?: string, };
|
export type PromptFormRequest = { id: string, title: string, description?: string, inputs: Array<FormInput>, confirmText?: string, cancelText?: string, size?: DialogSize, };
|
||||||
|
|
||||||
export type PromptFormResponse = { values: { [key in string]?: JsonPrimitive } | null, };
|
export type PromptFormResponse = { values: { [key in string]?: JsonPrimitive } | null, done?: boolean, };
|
||||||
|
|
||||||
export type PromptTextRequest = { id: string, title: string, label: string, description?: string, defaultValue?: string, placeholder?: string,
|
export type PromptTextRequest = { id: string, title: string, label: string, description?: string, defaultValue?: string, placeholder?: string,
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ export type HttpResponseEvent = { model: "http_response_event", id: string, crea
|
|||||||
* This mirrors `yaak_http::sender::HttpResponseEvent` but with serde support.
|
* This mirrors `yaak_http::sender::HttpResponseEvent` but with serde support.
|
||||||
* The `From` impl is in yaak-http to avoid circular dependencies.
|
* The `From` impl is in yaak-http to avoid circular dependencies.
|
||||||
*/
|
*/
|
||||||
export type HttpResponseEventData = { "type": "setting", name: string, value: string, } | { "type": "info", message: string, } | { "type": "redirect", url: string, status: number, behavior: string, } | { "type": "send_url", method: string, path: string, } | { "type": "receive_url", version: string, status: string, } | { "type": "header_up", name: string, value: string, } | { "type": "header_down", name: string, value: string, } | { "type": "chunk_sent", bytes: number, } | { "type": "chunk_received", bytes: number, } | { "type": "dns_resolved", hostname: string, addresses: Array<string>, duration: bigint, overridden: boolean, };
|
export type HttpResponseEventData = { "type": "setting", name: string, value: string, } | { "type": "info", message: string, } | { "type": "redirect", url: string, status: number, behavior: string, } | { "type": "send_url", method: string, scheme: string, username: string, password: string, host: string, port: number, path: string, query: string, fragment: string, } | { "type": "receive_url", version: string, status: string, } | { "type": "header_up", name: string, value: string, } | { "type": "header_down", name: string, value: string, } | { "type": "chunk_sent", bytes: number, } | { "type": "chunk_received", bytes: number, } | { "type": "dns_resolved", hostname: string, addresses: Array<string>, duration: bigint, overridden: boolean, };
|
||||||
|
|
||||||
export type HttpResponseHeader = { name: string, value: string, };
|
export type HttpResponseHeader = { name: string, value: string, };
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
import type {
|
import type {
|
||||||
FindHttpResponsesRequest,
|
FindHttpResponsesRequest,
|
||||||
FindHttpResponsesResponse,
|
FindHttpResponsesResponse,
|
||||||
|
FormInput,
|
||||||
GetCookieValueRequest,
|
GetCookieValueRequest,
|
||||||
GetCookieValueResponse,
|
GetCookieValueResponse,
|
||||||
GetHttpRequestByIdRequest,
|
GetHttpRequestByIdRequest,
|
||||||
GetHttpRequestByIdResponse,
|
GetHttpRequestByIdResponse,
|
||||||
|
JsonPrimitive,
|
||||||
ListCookieNamesResponse,
|
ListCookieNamesResponse,
|
||||||
ListFoldersRequest,
|
ListFoldersRequest,
|
||||||
ListFoldersResponse,
|
ListFoldersResponse,
|
||||||
@@ -27,6 +29,39 @@ import type {
|
|||||||
} from '../bindings/gen_events.ts';
|
} from '../bindings/gen_events.ts';
|
||||||
import type { Folder, HttpRequest } from '../bindings/gen_models.ts';
|
import type { Folder, HttpRequest } from '../bindings/gen_models.ts';
|
||||||
import type { JsonValue } from '../bindings/serde_json/JsonValue';
|
import type { JsonValue } from '../bindings/serde_json/JsonValue';
|
||||||
|
import type { MaybePromise } from '../helpers';
|
||||||
|
|
||||||
|
export type CallPromptFormDynamicArgs = {
|
||||||
|
values: { [key in string]?: JsonPrimitive };
|
||||||
|
};
|
||||||
|
|
||||||
|
type AddDynamicMethod<T> = {
|
||||||
|
dynamic?: (
|
||||||
|
ctx: Context,
|
||||||
|
args: CallPromptFormDynamicArgs,
|
||||||
|
) => MaybePromise<Partial<T> | null | undefined>;
|
||||||
|
};
|
||||||
|
|
||||||
|
// biome-ignore lint/suspicious/noExplicitAny: distributive conditional type pattern
|
||||||
|
type AddDynamic<T> = T extends any
|
||||||
|
? T extends { inputs?: FormInput[] }
|
||||||
|
? Omit<T, 'inputs'> & {
|
||||||
|
inputs: Array<AddDynamic<FormInput>>;
|
||||||
|
dynamic?: (
|
||||||
|
ctx: Context,
|
||||||
|
args: CallPromptFormDynamicArgs,
|
||||||
|
) => MaybePromise<
|
||||||
|
Partial<Omit<T, 'inputs'> & { inputs: Array<AddDynamic<FormInput>> }> | null | undefined
|
||||||
|
>;
|
||||||
|
}
|
||||||
|
: T & AddDynamicMethod<T>
|
||||||
|
: never;
|
||||||
|
|
||||||
|
export type DynamicPromptFormArg = AddDynamic<FormInput>;
|
||||||
|
|
||||||
|
type DynamicPromptFormRequest = Omit<PromptFormRequest, 'inputs'> & {
|
||||||
|
inputs: DynamicPromptFormArg[];
|
||||||
|
};
|
||||||
|
|
||||||
export type WorkspaceHandle = Pick<WorkspaceInfo, 'id' | 'name'>;
|
export type WorkspaceHandle = Pick<WorkspaceInfo, 'id' | 'name'>;
|
||||||
|
|
||||||
@@ -39,7 +74,7 @@ export interface Context {
|
|||||||
};
|
};
|
||||||
prompt: {
|
prompt: {
|
||||||
text(args: PromptTextRequest): Promise<PromptTextResponse['value']>;
|
text(args: PromptTextRequest): Promise<PromptTextResponse['value']>;
|
||||||
form(args: PromptFormRequest): Promise<PromptFormResponse['values']>;
|
form(args: DynamicPromptFormRequest): Promise<PromptFormResponse['values']>;
|
||||||
};
|
};
|
||||||
store: {
|
store: {
|
||||||
set<T>(key: string, value: T): Promise<void>;
|
set<T>(key: string, value: T): Promise<void>;
|
||||||
|
|||||||
@@ -2,21 +2,22 @@ import type { AuthenticationPlugin } from './AuthenticationPlugin';
|
|||||||
|
|
||||||
import type { Context } from './Context';
|
import type { Context } from './Context';
|
||||||
import type { FilterPlugin } from './FilterPlugin';
|
import type { FilterPlugin } from './FilterPlugin';
|
||||||
|
import type { FolderActionPlugin } from './FolderActionPlugin';
|
||||||
import type { GrpcRequestActionPlugin } from './GrpcRequestActionPlugin';
|
import type { GrpcRequestActionPlugin } from './GrpcRequestActionPlugin';
|
||||||
import type { HttpRequestActionPlugin } from './HttpRequestActionPlugin';
|
import type { HttpRequestActionPlugin } from './HttpRequestActionPlugin';
|
||||||
import type { WebsocketRequestActionPlugin } from './WebsocketRequestActionPlugin';
|
|
||||||
import type { WorkspaceActionPlugin } from './WorkspaceActionPlugin';
|
|
||||||
import type { FolderActionPlugin } from './FolderActionPlugin';
|
|
||||||
import type { ImporterPlugin } from './ImporterPlugin';
|
import type { ImporterPlugin } from './ImporterPlugin';
|
||||||
import type { TemplateFunctionPlugin } from './TemplateFunctionPlugin';
|
import type { TemplateFunctionPlugin } from './TemplateFunctionPlugin';
|
||||||
import type { ThemePlugin } from './ThemePlugin';
|
import type { ThemePlugin } from './ThemePlugin';
|
||||||
|
import type { WebsocketRequestActionPlugin } from './WebsocketRequestActionPlugin';
|
||||||
|
import type { WorkspaceActionPlugin } from './WorkspaceActionPlugin';
|
||||||
|
|
||||||
export type { Context };
|
export type { Context };
|
||||||
export type { DynamicTemplateFunctionArg } from './TemplateFunctionPlugin';
|
|
||||||
export type { DynamicAuthenticationArg } from './AuthenticationPlugin';
|
export type { DynamicAuthenticationArg } from './AuthenticationPlugin';
|
||||||
|
export type { CallPromptFormDynamicArgs, DynamicPromptFormArg } from './Context';
|
||||||
|
export type { DynamicTemplateFunctionArg } from './TemplateFunctionPlugin';
|
||||||
export type { TemplateFunctionPlugin };
|
export type { TemplateFunctionPlugin };
|
||||||
export type { WorkspaceActionPlugin } from './WorkspaceActionPlugin';
|
|
||||||
export type { FolderActionPlugin } from './FolderActionPlugin';
|
export type { FolderActionPlugin } from './FolderActionPlugin';
|
||||||
|
export type { WorkspaceActionPlugin } from './WorkspaceActionPlugin';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The global structure of a Yaak plugin
|
* The global structure of a Yaak plugin
|
||||||
|
|||||||
@@ -1,7 +1,12 @@
|
|||||||
import console from 'node:console';
|
import console from 'node:console';
|
||||||
import { type Stats, statSync, watch } from 'node:fs';
|
import { type Stats, statSync, watch } from 'node:fs';
|
||||||
import path from 'node:path';
|
import path from 'node:path';
|
||||||
import type { Context, PluginDefinition } from '@yaakapp/api';
|
import type {
|
||||||
|
CallPromptFormDynamicArgs,
|
||||||
|
Context,
|
||||||
|
DynamicPromptFormArg,
|
||||||
|
PluginDefinition,
|
||||||
|
} from '@yaakapp/api';
|
||||||
import {
|
import {
|
||||||
applyFormInputDefaults,
|
applyFormInputDefaults,
|
||||||
validateTemplateFunctionArgs,
|
validateTemplateFunctionArgs,
|
||||||
@@ -12,6 +17,7 @@ import type {
|
|||||||
DeleteModelResponse,
|
DeleteModelResponse,
|
||||||
FindHttpResponsesResponse,
|
FindHttpResponsesResponse,
|
||||||
Folder,
|
Folder,
|
||||||
|
FormInput,
|
||||||
GetCookieValueRequest,
|
GetCookieValueRequest,
|
||||||
GetCookieValueResponse,
|
GetCookieValueResponse,
|
||||||
GetHttpRequestByIdResponse,
|
GetHttpRequestByIdResponse,
|
||||||
@@ -55,6 +61,7 @@ export class PluginInstance {
|
|||||||
#mod: PluginDefinition;
|
#mod: PluginDefinition;
|
||||||
#pluginToAppEvents: EventChannel;
|
#pluginToAppEvents: EventChannel;
|
||||||
#appToPluginEvents: EventChannel;
|
#appToPluginEvents: EventChannel;
|
||||||
|
#pendingDynamicForms = new Map<string, DynamicPromptFormArg[]>();
|
||||||
|
|
||||||
constructor(workerData: PluginWorkerData, pluginEvents: EventChannel) {
|
constructor(workerData: PluginWorkerData, pluginEvents: EventChannel) {
|
||||||
this.#workerData = workerData;
|
this.#workerData = workerData;
|
||||||
@@ -106,6 +113,7 @@ export class PluginInstance {
|
|||||||
|
|
||||||
async terminate() {
|
async terminate() {
|
||||||
await this.#mod?.dispose?.();
|
await this.#mod?.dispose?.();
|
||||||
|
this.#pendingDynamicForms.clear();
|
||||||
this.#unimportModule();
|
this.#unimportModule();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -299,7 +307,7 @@ export class PluginInstance {
|
|||||||
const replyPayload: InternalEventPayload = {
|
const replyPayload: InternalEventPayload = {
|
||||||
type: 'get_template_function_config_response',
|
type: 'get_template_function_config_response',
|
||||||
pluginRefId: this.#workerData.pluginRefId,
|
pluginRefId: this.#workerData.pluginRefId,
|
||||||
function: { ...fn, args: resolvedArgs },
|
function: { ...fn, args: stripDynamicCallbacks(resolvedArgs) },
|
||||||
};
|
};
|
||||||
this.#sendPayload(context, replyPayload, replyId);
|
this.#sendPayload(context, replyPayload, replyId);
|
||||||
return;
|
return;
|
||||||
@@ -326,7 +334,7 @@ export class PluginInstance {
|
|||||||
|
|
||||||
const replyPayload: InternalEventPayload = {
|
const replyPayload: InternalEventPayload = {
|
||||||
type: 'get_http_authentication_config_response',
|
type: 'get_http_authentication_config_response',
|
||||||
args: resolvedArgs,
|
args: stripDynamicCallbacks(resolvedArgs),
|
||||||
actions: resolvedActions,
|
actions: resolvedActions,
|
||||||
pluginRefId: this.#workerData.pluginRefId,
|
pluginRefId: this.#workerData.pluginRefId,
|
||||||
};
|
};
|
||||||
@@ -664,10 +672,66 @@ export class PluginInstance {
|
|||||||
return reply.value;
|
return reply.value;
|
||||||
},
|
},
|
||||||
form: async (args) => {
|
form: async (args) => {
|
||||||
const reply: PromptFormResponse = await this.#sendForReply(context, {
|
// Resolve dynamic callbacks on initial inputs using default values
|
||||||
type: 'prompt_form_request',
|
const defaults = applyFormInputDefaults(args.inputs, {});
|
||||||
...args,
|
const callArgs: CallPromptFormDynamicArgs = { values: defaults };
|
||||||
|
const resolvedInputs = await applyDynamicFormInput(
|
||||||
|
this.#newCtx(context),
|
||||||
|
args.inputs,
|
||||||
|
callArgs,
|
||||||
|
);
|
||||||
|
const strippedInputs = stripDynamicCallbacks(resolvedInputs);
|
||||||
|
|
||||||
|
// Build the event manually so we can get the event ID for keying
|
||||||
|
const eventToSend = this.#buildEventToSend(
|
||||||
|
context,
|
||||||
|
{ type: 'prompt_form_request', ...args, inputs: strippedInputs },
|
||||||
|
null,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Store original inputs (with dynamic callbacks) for later resolution
|
||||||
|
this.#pendingDynamicForms.set(eventToSend.id, args.inputs);
|
||||||
|
|
||||||
|
const reply = await new Promise<PromptFormResponse>((resolve) => {
|
||||||
|
const cb = (event: InternalEvent) => {
|
||||||
|
if (event.replyId !== eventToSend.id) return;
|
||||||
|
|
||||||
|
if (event.payload.type === 'prompt_form_response') {
|
||||||
|
const { done, values } = event.payload as PromptFormResponse;
|
||||||
|
if (done) {
|
||||||
|
// Final response — resolve the promise and clean up
|
||||||
|
this.#appToPluginEvents.unlisten(cb);
|
||||||
|
this.#pendingDynamicForms.delete(eventToSend.id);
|
||||||
|
resolve({ values } as PromptFormResponse);
|
||||||
|
} else {
|
||||||
|
// Intermediate value change — resolve dynamic inputs and send back
|
||||||
|
// Skip empty values (fired on initial mount before user interaction)
|
||||||
|
const storedInputs = this.#pendingDynamicForms.get(eventToSend.id);
|
||||||
|
if (storedInputs && values && Object.keys(values).length > 0) {
|
||||||
|
const ctx = this.#newCtx(context);
|
||||||
|
const callArgs: CallPromptFormDynamicArgs = { values };
|
||||||
|
applyDynamicFormInput(ctx, storedInputs, callArgs)
|
||||||
|
.then((resolvedInputs) => {
|
||||||
|
const stripped = stripDynamicCallbacks(resolvedInputs);
|
||||||
|
this.#sendPayload(
|
||||||
|
context,
|
||||||
|
{ type: 'prompt_form_request', ...args, inputs: stripped },
|
||||||
|
eventToSend.id,
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
console.error('Failed to resolve dynamic form inputs', err);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
this.#appToPluginEvents.listen(cb);
|
||||||
|
|
||||||
|
// Send the initial event after we start listening (to prevent race)
|
||||||
|
this.#sendEvent(eventToSend);
|
||||||
});
|
});
|
||||||
|
|
||||||
return reply.values;
|
return reply.values;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -788,12 +852,12 @@ export class PluginInstance {
|
|||||||
const { folders } = await this.#sendForReply<ListFoldersResponse>(context, payload);
|
const { folders } = await this.#sendForReply<ListFoldersResponse>(context, payload);
|
||||||
return folders.find((f) => f.id === args.id) ?? null;
|
return folders.find((f) => f.id === args.id) ?? null;
|
||||||
},
|
},
|
||||||
create: async (args) => {
|
create: async ({ name, ...args }) => {
|
||||||
const payload = {
|
const payload = {
|
||||||
type: 'upsert_model_request',
|
type: 'upsert_model_request',
|
||||||
model: {
|
model: {
|
||||||
name: '',
|
|
||||||
...args,
|
...args,
|
||||||
|
name: name ?? '',
|
||||||
id: '',
|
id: '',
|
||||||
model: 'folder',
|
model: 'folder',
|
||||||
},
|
},
|
||||||
@@ -906,6 +970,17 @@ export class PluginInstance {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function stripDynamicCallbacks(inputs: { dynamic?: unknown }[]): FormInput[] {
|
||||||
|
return inputs.map((input) => {
|
||||||
|
// biome-ignore lint/suspicious/noExplicitAny: stripping dynamic from union type
|
||||||
|
const { dynamic, ...rest } = input as any;
|
||||||
|
if ('inputs' in rest && Array.isArray(rest.inputs)) {
|
||||||
|
rest.inputs = stripDynamicCallbacks(rest.inputs);
|
||||||
|
}
|
||||||
|
return rest as FormInput;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function genId(len = 5): string {
|
function genId(len = 5): string {
|
||||||
const alphabet = '01234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
|
const alphabet = '01234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
|
||||||
let id = '';
|
let id = '';
|
||||||
|
|||||||
@@ -1,9 +1,21 @@
|
|||||||
import type { Context, DynamicAuthenticationArg, DynamicTemplateFunctionArg } from '@yaakapp/api';
|
import type {
|
||||||
|
CallPromptFormDynamicArgs,
|
||||||
|
Context,
|
||||||
|
DynamicAuthenticationArg,
|
||||||
|
DynamicPromptFormArg,
|
||||||
|
DynamicTemplateFunctionArg,
|
||||||
|
} from '@yaakapp/api';
|
||||||
import type {
|
import type {
|
||||||
CallHttpAuthenticationActionArgs,
|
CallHttpAuthenticationActionArgs,
|
||||||
CallTemplateFunctionArgs,
|
CallTemplateFunctionArgs,
|
||||||
} from '@yaakapp-internal/plugins';
|
} from '@yaakapp-internal/plugins';
|
||||||
|
|
||||||
|
type AnyDynamicArg = DynamicTemplateFunctionArg | DynamicAuthenticationArg | DynamicPromptFormArg;
|
||||||
|
type AnyCallArgs =
|
||||||
|
| CallTemplateFunctionArgs
|
||||||
|
| CallHttpAuthenticationActionArgs
|
||||||
|
| CallPromptFormDynamicArgs;
|
||||||
|
|
||||||
export async function applyDynamicFormInput(
|
export async function applyDynamicFormInput(
|
||||||
ctx: Context,
|
ctx: Context,
|
||||||
args: DynamicTemplateFunctionArg[],
|
args: DynamicTemplateFunctionArg[],
|
||||||
@@ -18,30 +30,40 @@ export async function applyDynamicFormInput(
|
|||||||
|
|
||||||
export async function applyDynamicFormInput(
|
export async function applyDynamicFormInput(
|
||||||
ctx: Context,
|
ctx: Context,
|
||||||
args: (DynamicTemplateFunctionArg | DynamicAuthenticationArg)[],
|
args: DynamicPromptFormArg[],
|
||||||
callArgs: CallTemplateFunctionArgs | CallHttpAuthenticationActionArgs,
|
callArgs: CallPromptFormDynamicArgs,
|
||||||
): Promise<(DynamicTemplateFunctionArg | DynamicAuthenticationArg)[]> {
|
): Promise<DynamicPromptFormArg[]>;
|
||||||
const resolvedArgs: (DynamicTemplateFunctionArg | DynamicAuthenticationArg)[] = [];
|
|
||||||
|
export async function applyDynamicFormInput(
|
||||||
|
ctx: Context,
|
||||||
|
args: AnyDynamicArg[],
|
||||||
|
callArgs: AnyCallArgs,
|
||||||
|
): Promise<AnyDynamicArg[]> {
|
||||||
|
const resolvedArgs: AnyDynamicArg[] = [];
|
||||||
for (const { dynamic, ...arg } of args) {
|
for (const { dynamic, ...arg } of args) {
|
||||||
const dynamicResult =
|
const dynamicResult =
|
||||||
typeof dynamic === 'function'
|
typeof dynamic === 'function'
|
||||||
? await dynamic(
|
? await dynamic(
|
||||||
ctx,
|
ctx,
|
||||||
callArgs as CallTemplateFunctionArgs & CallHttpAuthenticationActionArgs,
|
callArgs as CallTemplateFunctionArgs &
|
||||||
|
CallHttpAuthenticationActionArgs &
|
||||||
|
CallPromptFormDynamicArgs,
|
||||||
)
|
)
|
||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
const newArg = {
|
const newArg = {
|
||||||
...arg,
|
...arg,
|
||||||
...dynamicResult,
|
...dynamicResult,
|
||||||
} as DynamicTemplateFunctionArg | DynamicAuthenticationArg;
|
} as AnyDynamicArg;
|
||||||
|
|
||||||
if ('inputs' in newArg && Array.isArray(newArg.inputs)) {
|
if ('inputs' in newArg && Array.isArray(newArg.inputs)) {
|
||||||
try {
|
try {
|
||||||
newArg.inputs = await applyDynamicFormInput(
|
newArg.inputs = await applyDynamicFormInput(
|
||||||
ctx,
|
ctx,
|
||||||
newArg.inputs as DynamicTemplateFunctionArg[],
|
newArg.inputs as DynamicTemplateFunctionArg[],
|
||||||
callArgs as CallTemplateFunctionArgs & CallHttpAuthenticationActionArgs,
|
callArgs as CallTemplateFunctionArgs &
|
||||||
|
CallHttpAuthenticationActionArgs &
|
||||||
|
CallPromptFormDynamicArgs,
|
||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('Failed to apply dynamic form input', e);
|
console.error('Failed to apply dynamic form input', e);
|
||||||
|
|||||||
45
plugins-external/httpsnippet/README.md
Normal file
45
plugins-external/httpsnippet/README.md
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
# Yaak HTTP Snippet Plugin
|
||||||
|
|
||||||
|
Generate code snippets for HTTP requests in various languages and frameworks,
|
||||||
|
powered by [@readme/httpsnippet](https://github.com/readmeio/httpsnippet).
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## How It Works
|
||||||
|
|
||||||
|
Right-click any HTTP request (or use the `...` menu) and select **Generate Code Snippet**.
|
||||||
|
A dialog lets you pick a language and library, with a live preview of the generated code.
|
||||||
|
Click **Copy to Clipboard** to copy the snippet. Your language and library selections are
|
||||||
|
remembered for next time.
|
||||||
|
|
||||||
|
## Supported Languages
|
||||||
|
|
||||||
|
Each language supports one or more libraries:
|
||||||
|
|
||||||
|
| Language | Libraries |
|
||||||
|
|---|---|
|
||||||
|
| C | libcurl |
|
||||||
|
| Clojure | clj-http |
|
||||||
|
| C# | HttpClient, RestSharp |
|
||||||
|
| Go | Native |
|
||||||
|
| HTTP | HTTP/1.1 |
|
||||||
|
| Java | AsyncHttp, NetHttp, OkHttp, Unirest |
|
||||||
|
| JavaScript | Axios, fetch, jQuery, XHR |
|
||||||
|
| Kotlin | OkHttp |
|
||||||
|
| Node.js | Axios, fetch, HTTP, Request, Unirest |
|
||||||
|
| Objective-C | NSURLSession |
|
||||||
|
| OCaml | CoHTTP |
|
||||||
|
| PHP | cURL, Guzzle, HTTP v1, HTTP v2 |
|
||||||
|
| PowerShell | Invoke-WebRequest, RestMethod |
|
||||||
|
| Python | http.client, Requests |
|
||||||
|
| R | httr |
|
||||||
|
| Ruby | Native |
|
||||||
|
| Shell | cURL, HTTPie, Wget |
|
||||||
|
| Swift | URLSession |
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
- Renders template variables before generating snippets, so the output reflects real values
|
||||||
|
- Supports all body types: JSON, form-urlencoded, multipart, GraphQL, and raw text
|
||||||
|
- Includes authentication headers (Basic, Bearer, and API Key)
|
||||||
|
- Includes query parameters and custom headers
|
||||||
24
plugins-external/httpsnippet/package.json
Normal file
24
plugins-external/httpsnippet/package.json
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
"name": "@yaak/httpsnippet",
|
||||||
|
"private": true,
|
||||||
|
"version": "1.0.3",
|
||||||
|
"displayName": "HTTP Snippet",
|
||||||
|
"description": "Generate code snippets for HTTP requests in various languages and frameworks",
|
||||||
|
"minYaakVersion": "2026.2.0-beta.10",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/mountain-loop/yaak.git",
|
||||||
|
"directory": "plugins-external/httpsnippet"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"build": "yaakcli build",
|
||||||
|
"dev": "yaakcli dev"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@readme/httpsnippet": "^11.0.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/node": "^22.0.0",
|
||||||
|
"typescript": "^5.9.3"
|
||||||
|
}
|
||||||
|
}
|
||||||
314
plugins-external/httpsnippet/src/index.ts
Normal file
314
plugins-external/httpsnippet/src/index.ts
Normal file
@@ -0,0 +1,314 @@
|
|||||||
|
import { availableTargets, type HarRequest, HTTPSnippet } from '@readme/httpsnippet';
|
||||||
|
import type { EditorLanguage, HttpRequest, PluginDefinition } from '@yaakapp/api';
|
||||||
|
|
||||||
|
// Get all available targets and build select options
|
||||||
|
const targets = availableTargets();
|
||||||
|
|
||||||
|
// Targets to exclude from the language list
|
||||||
|
const excludedTargets = new Set(['json']);
|
||||||
|
|
||||||
|
// Build language (target) options
|
||||||
|
const languageOptions = targets
|
||||||
|
.filter((target) => !excludedTargets.has(target.key))
|
||||||
|
.map((target) => ({
|
||||||
|
label: target.title,
|
||||||
|
value: target.key,
|
||||||
|
}));
|
||||||
|
|
||||||
|
// Preferred clients per target (shown first in the list)
|
||||||
|
const preferredClients: Record<string, string> = {
|
||||||
|
javascript: 'fetch',
|
||||||
|
node: 'fetch',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Get client options for a given target key
|
||||||
|
function getClientOptions(targetKey: string) {
|
||||||
|
const target = targets.find((t) => t.key === targetKey);
|
||||||
|
if (!target) return [];
|
||||||
|
const preferred = preferredClients[targetKey];
|
||||||
|
return target.clients
|
||||||
|
.map((client) => ({
|
||||||
|
label: client.title,
|
||||||
|
value: client.key,
|
||||||
|
}))
|
||||||
|
.sort((a, b) => {
|
||||||
|
if (a.value === preferred) return -1;
|
||||||
|
if (b.value === preferred) return 1;
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get default client for a target
|
||||||
|
function getDefaultClient(targetKey: string): string {
|
||||||
|
const options = getClientOptions(targetKey);
|
||||||
|
return options[0]?.value ?? '';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Defaults
|
||||||
|
const defaultTarget = 'javascript';
|
||||||
|
|
||||||
|
// Map httpsnippet target key to editor language for syntax highlighting
|
||||||
|
const editorLanguageMap: Record<string, EditorLanguage> = {
|
||||||
|
c: 'c',
|
||||||
|
clojure: 'clojure',
|
||||||
|
csharp: 'csharp',
|
||||||
|
go: 'go',
|
||||||
|
http: 'http',
|
||||||
|
java: 'java',
|
||||||
|
javascript: 'javascript',
|
||||||
|
kotlin: 'kotlin',
|
||||||
|
node: 'javascript',
|
||||||
|
objc: 'objective_c',
|
||||||
|
ocaml: 'ocaml',
|
||||||
|
php: 'php',
|
||||||
|
powershell: 'powershell',
|
||||||
|
python: 'python',
|
||||||
|
r: 'r',
|
||||||
|
ruby: 'ruby',
|
||||||
|
shell: 'shell',
|
||||||
|
swift: 'swift',
|
||||||
|
};
|
||||||
|
|
||||||
|
function getEditorLanguage(targetKey: string): EditorLanguage {
|
||||||
|
return editorLanguageMap[targetKey] ?? 'text';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert Yaak HttpRequest to HAR format
|
||||||
|
function toHarRequest(request: Partial<HttpRequest>) {
|
||||||
|
// Build URL with query parameters
|
||||||
|
let finalUrl = request.url || '';
|
||||||
|
const urlParams = (request.urlParameters ?? []).filter((p) => p.enabled !== false && !!p.name);
|
||||||
|
if (urlParams.length > 0) {
|
||||||
|
const [base, hash] = finalUrl.split('#');
|
||||||
|
const separator = base?.includes('?') ? '&' : '?';
|
||||||
|
const queryString = urlParams
|
||||||
|
.map((p) => `${encodeURIComponent(p.name)}=${encodeURIComponent(p.value)}`)
|
||||||
|
.join('&');
|
||||||
|
finalUrl = base + separator + queryString + (hash ? `#${hash}` : '');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build headers array
|
||||||
|
const headers: Array<{ name: string; value: string }> = (request.headers ?? [])
|
||||||
|
.filter((h) => h.enabled !== false && !!h.name)
|
||||||
|
.map((h) => ({ name: h.name, value: h.value }));
|
||||||
|
|
||||||
|
// Handle authentication
|
||||||
|
if (request.authentication?.disabled !== true) {
|
||||||
|
if (request.authenticationType === 'basic') {
|
||||||
|
const credentials = btoa(
|
||||||
|
`${request.authentication?.username ?? ''}:${request.authentication?.password ?? ''}`,
|
||||||
|
);
|
||||||
|
headers.push({ name: 'Authorization', value: `Basic ${credentials}` });
|
||||||
|
} else if (request.authenticationType === 'bearer') {
|
||||||
|
const prefix = request.authentication?.prefix ?? 'Bearer';
|
||||||
|
const token = request.authentication?.token ?? '';
|
||||||
|
headers.push({ name: 'Authorization', value: `${prefix} ${token}`.trim() });
|
||||||
|
} else if (request.authenticationType === 'apikey') {
|
||||||
|
if (request.authentication?.location === 'header') {
|
||||||
|
headers.push({
|
||||||
|
name: request.authentication?.key ?? 'X-Api-Key',
|
||||||
|
value: request.authentication?.value ?? '',
|
||||||
|
});
|
||||||
|
} else if (request.authentication?.location === 'query') {
|
||||||
|
const sep = finalUrl.includes('?') ? '&' : '?';
|
||||||
|
finalUrl = [
|
||||||
|
finalUrl,
|
||||||
|
sep,
|
||||||
|
encodeURIComponent(request.authentication?.key ?? 'token'),
|
||||||
|
'=',
|
||||||
|
encodeURIComponent(request.authentication?.value ?? ''),
|
||||||
|
].join('');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build HAR request object
|
||||||
|
const har: Record<string, unknown> = {
|
||||||
|
method: request.method || 'GET',
|
||||||
|
url: finalUrl,
|
||||||
|
headers,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Handle request body
|
||||||
|
const bodyType = request.bodyType ?? 'none';
|
||||||
|
if (bodyType !== 'none' && request.body) {
|
||||||
|
if (bodyType === 'application/x-www-form-urlencoded' && Array.isArray(request.body.form)) {
|
||||||
|
const params = request.body.form
|
||||||
|
.filter((p: { enabled?: boolean; name?: string }) => p.enabled !== false && !!p.name)
|
||||||
|
.map((p: { name: string; value: string }) => ({ name: p.name, value: p.value }));
|
||||||
|
har.postData = {
|
||||||
|
mimeType: 'application/x-www-form-urlencoded',
|
||||||
|
params,
|
||||||
|
};
|
||||||
|
} else if (bodyType === 'multipart/form-data' && Array.isArray(request.body.form)) {
|
||||||
|
const params = request.body.form
|
||||||
|
.filter((p: { enabled?: boolean; name?: string }) => p.enabled !== false && !!p.name)
|
||||||
|
.map((p: { name: string; value: string; file?: string; contentType?: string }) => {
|
||||||
|
const param: Record<string, string> = { name: p.name, value: p.value || '' };
|
||||||
|
if (p.file) param.fileName = p.file;
|
||||||
|
if (p.contentType) param.contentType = p.contentType;
|
||||||
|
return param;
|
||||||
|
});
|
||||||
|
har.postData = {
|
||||||
|
mimeType: 'multipart/form-data',
|
||||||
|
params,
|
||||||
|
};
|
||||||
|
} else if (bodyType === 'graphql' && typeof request.body.query === 'string') {
|
||||||
|
const body = {
|
||||||
|
query: request.body.query || '',
|
||||||
|
variables: maybeParseJSON(request.body.variables, undefined),
|
||||||
|
};
|
||||||
|
har.postData = {
|
||||||
|
mimeType: 'application/json',
|
||||||
|
text: JSON.stringify(body),
|
||||||
|
};
|
||||||
|
} else if (typeof request.body.text === 'string') {
|
||||||
|
har.postData = {
|
||||||
|
mimeType: bodyType,
|
||||||
|
text: request.body.text,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return har;
|
||||||
|
}
|
||||||
|
|
||||||
|
function maybeParseJSON<T>(v: unknown, fallback: T): T | unknown {
|
||||||
|
if (typeof v !== 'string') return fallback;
|
||||||
|
try {
|
||||||
|
return JSON.parse(v);
|
||||||
|
} catch {
|
||||||
|
return fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const plugin: PluginDefinition = {
|
||||||
|
httpRequestActions: [
|
||||||
|
{
|
||||||
|
label: 'Generate Code Snippet',
|
||||||
|
icon: 'copy',
|
||||||
|
async onSelect(ctx, args) {
|
||||||
|
// Render the request with variables resolved
|
||||||
|
const renderedRequest = await ctx.httpRequest.render({
|
||||||
|
httpRequest: args.httpRequest,
|
||||||
|
purpose: 'send',
|
||||||
|
});
|
||||||
|
|
||||||
|
// Convert to HAR format
|
||||||
|
const harRequest = toHarRequest(renderedRequest) as HarRequest;
|
||||||
|
|
||||||
|
// Get previously selected language or use defaults
|
||||||
|
const storedTarget = await ctx.store.get<string>('selectedTarget');
|
||||||
|
const initialTarget = storedTarget || defaultTarget;
|
||||||
|
const storedClient = await ctx.store.get<string>(`selectedClient:${initialTarget}`);
|
||||||
|
const initialClient = storedClient || getDefaultClient(initialTarget);
|
||||||
|
|
||||||
|
// Create snippet generator
|
||||||
|
const snippet = new HTTPSnippet(harRequest);
|
||||||
|
const generateSnippet = (target: string, client: string): string => {
|
||||||
|
const result = snippet.convert(target as any, client);
|
||||||
|
return (Array.isArray(result) ? result.join('\n') : result || '').replace(/\r\n/g, '\n');
|
||||||
|
};
|
||||||
|
|
||||||
|
// Generate initial code preview
|
||||||
|
let initialCode = '';
|
||||||
|
try {
|
||||||
|
initialCode = generateSnippet(initialTarget, initialClient);
|
||||||
|
} catch {
|
||||||
|
initialCode = '// Error generating snippet';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show dialog with language/library selectors and code preview
|
||||||
|
const result = await ctx.prompt.form({
|
||||||
|
id: 'httpsnippet',
|
||||||
|
title: 'Generate Code Snippet',
|
||||||
|
confirmText: 'Copy to Clipboard',
|
||||||
|
cancelText: 'Cancel',
|
||||||
|
size: 'md',
|
||||||
|
inputs: [
|
||||||
|
{
|
||||||
|
type: 'h_stack',
|
||||||
|
inputs: [
|
||||||
|
{
|
||||||
|
type: 'select',
|
||||||
|
name: 'target',
|
||||||
|
label: 'Language',
|
||||||
|
defaultValue: initialTarget,
|
||||||
|
options: languageOptions,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'select',
|
||||||
|
name: `client-${initialTarget}`,
|
||||||
|
label: 'Library',
|
||||||
|
defaultValue: initialClient,
|
||||||
|
options: getClientOptions(initialTarget),
|
||||||
|
dynamic(_ctx, { values }) {
|
||||||
|
const targetKey = String(values.target || defaultTarget);
|
||||||
|
const options = getClientOptions(targetKey);
|
||||||
|
return {
|
||||||
|
name: `client-${targetKey}`,
|
||||||
|
options,
|
||||||
|
defaultValue: options[0]?.value ?? '',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'editor',
|
||||||
|
name: 'code',
|
||||||
|
label: 'Preview',
|
||||||
|
language: getEditorLanguage(initialTarget),
|
||||||
|
defaultValue: initialCode,
|
||||||
|
readOnly: true,
|
||||||
|
rows: 15,
|
||||||
|
dynamic(_ctx, { values }) {
|
||||||
|
const targetKey = String(values.target || defaultTarget);
|
||||||
|
const clientKey = String(
|
||||||
|
values[`client-${targetKey}`] || getDefaultClient(targetKey),
|
||||||
|
);
|
||||||
|
let code: string;
|
||||||
|
try {
|
||||||
|
code = generateSnippet(targetKey, clientKey);
|
||||||
|
} catch {
|
||||||
|
code = '// Error generating snippet';
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
defaultValue: code,
|
||||||
|
language: getEditorLanguage(targetKey),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
if (result) {
|
||||||
|
// Store the selected language and library for next time
|
||||||
|
const selectedTarget = String(result.target || initialTarget);
|
||||||
|
const selectedClient = String(
|
||||||
|
result[`client-${selectedTarget}`] || getDefaultClient(selectedTarget),
|
||||||
|
);
|
||||||
|
await ctx.store.set('selectedTarget', selectedTarget);
|
||||||
|
await ctx.store.set(`selectedClient:${selectedTarget}`, selectedClient);
|
||||||
|
|
||||||
|
// Generate snippet for the selected language
|
||||||
|
try {
|
||||||
|
const codeText = generateSnippet(selectedTarget, selectedClient);
|
||||||
|
await ctx.clipboard.copyText(codeText);
|
||||||
|
await ctx.toast.show({
|
||||||
|
message: 'Code snippet copied to clipboard',
|
||||||
|
icon: 'copy',
|
||||||
|
color: 'success',
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
await ctx.toast.show({
|
||||||
|
message: `Failed to generate snippet: ${err}`,
|
||||||
|
icon: 'alert_triangle',
|
||||||
|
color: 'danger',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
{
|
{
|
||||||
"name": "@yaak/mcp-server",
|
"name": "@yaak/mcp-server",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.1.7",
|
"version": "0.2.1",
|
||||||
"displayName": "MCP Server",
|
"displayName": "MCP Server",
|
||||||
"description": "Expose Yaak functionality via Model Context Protocol",
|
"description": "Expose Yaak functionality via Model Context Protocol",
|
||||||
"minYaakVersion": "2025.10.0-beta.6",
|
"minYaakVersion": "2026.1.0",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/mountain-loop/yaak.git",
|
"url": "https://github.com/mountain-loop/yaak.git",
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@hono/mcp": "^0.2.3",
|
"@hono/mcp": "^0.2.3",
|
||||||
"@hono/node-server": "^1.19.7",
|
"@hono/node-server": "^1.19.7",
|
||||||
"@modelcontextprotocol/sdk": "^1.25.2",
|
"@modelcontextprotocol/sdk": "^1.26.0",
|
||||||
"hono": "^4.11.7",
|
"hono": "^4.11.7",
|
||||||
"zod": "^3.25.76"
|
"zod": "^3.25.76"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ export const plugin: PluginDefinition = {
|
|||||||
async onSelect(ctx, args) {
|
async onSelect(ctx, args) {
|
||||||
const rendered_request = await ctx.httpRequest.render({
|
const rendered_request = await ctx.httpRequest.render({
|
||||||
httpRequest: args.httpRequest,
|
httpRequest: args.httpRequest,
|
||||||
purpose: 'preview',
|
purpose: 'send',
|
||||||
});
|
});
|
||||||
const data = await convertToCurl(rendered_request);
|
const data = await convertToCurl(rendered_request);
|
||||||
await ctx.clipboard.copyText(data);
|
await ctx.clipboard.copyText(data);
|
||||||
|
|||||||
@@ -184,6 +184,18 @@ export function buildHostedCallbackRedirectUri(localPort: number, localPath: str
|
|||||||
return `${HOSTED_CALLBACK_URL}?redirect_to=${encodeURIComponent(localRedirectUri)}`;
|
return `${HOSTED_CALLBACK_URL}?redirect_to=${encodeURIComponent(localRedirectUri)}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop the active callback server if one is running.
|
||||||
|
* Called during plugin dispose to ensure the server is cleaned up before the process exits.
|
||||||
|
*/
|
||||||
|
export function stopActiveServer(): void {
|
||||||
|
if (activeServer) {
|
||||||
|
console.log('[oauth2] Stopping active callback server during dispose');
|
||||||
|
activeServer.stop();
|
||||||
|
activeServer = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Open an authorization URL in the system browser, start a local callback server,
|
* Open an authorization URL in the system browser, start a local callback server,
|
||||||
* and wait for the OAuth provider to redirect back.
|
* and wait for the OAuth provider to redirect back.
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import type {
|
|||||||
JsonPrimitive,
|
JsonPrimitive,
|
||||||
PluginDefinition,
|
PluginDefinition,
|
||||||
} from '@yaakapp/api';
|
} from '@yaakapp/api';
|
||||||
import { DEFAULT_LOCALHOST_PORT, HOSTED_CALLBACK_URL } from './callbackServer';
|
import { DEFAULT_LOCALHOST_PORT, HOSTED_CALLBACK_URL, stopActiveServer } from './callbackServer';
|
||||||
import {
|
import {
|
||||||
type CallbackType,
|
type CallbackType,
|
||||||
DEFAULT_PKCE_METHOD,
|
DEFAULT_PKCE_METHOD,
|
||||||
@@ -78,6 +78,9 @@ const accessTokenUrls = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
export const plugin: PluginDefinition = {
|
export const plugin: PluginDefinition = {
|
||||||
|
dispose() {
|
||||||
|
stopActiveServer();
|
||||||
|
},
|
||||||
authentication: {
|
authentication: {
|
||||||
name: 'oauth2',
|
name: 'oauth2',
|
||||||
label: 'OAuth 2.0',
|
label: 'OAuth 2.0',
|
||||||
|
|||||||
@@ -55,6 +55,7 @@ const requestArg: FormInput = {
|
|||||||
type: 'http_request',
|
type: 'http_request',
|
||||||
name: 'request',
|
name: 'request',
|
||||||
label: 'Request',
|
label: 'Request',
|
||||||
|
defaultValue: '', // Make it not select the active one by default
|
||||||
};
|
};
|
||||||
|
|
||||||
export const plugin: PluginDefinition = {
|
export const plugin: PluginDefinition = {
|
||||||
|
|||||||
@@ -1,21 +1,14 @@
|
|||||||
import { getModel } from '@yaakapp-internal/models';
|
import { getModel } from '@yaakapp-internal/models';
|
||||||
import { Icon } from '../components/core/Icon';
|
|
||||||
import { HStack } from '../components/core/Stacks';
|
|
||||||
import type { FolderSettingsTab } from '../components/FolderSettingsDialog';
|
import type { FolderSettingsTab } from '../components/FolderSettingsDialog';
|
||||||
import { FolderSettingsDialog } from '../components/FolderSettingsDialog';
|
import { FolderSettingsDialog } from '../components/FolderSettingsDialog';
|
||||||
import { showDialog } from '../lib/dialog';
|
import { showDialog } from '../lib/dialog';
|
||||||
import { resolvedModelName } from '../lib/resolvedModelName';
|
|
||||||
|
|
||||||
export function openFolderSettings(folderId: string, tab?: FolderSettingsTab) {
|
export function openFolderSettings(folderId: string, tab?: FolderSettingsTab) {
|
||||||
const folder = getModel('folder', folderId);
|
const folder = getModel('folder', folderId);
|
||||||
|
if (folder == null) return;
|
||||||
showDialog({
|
showDialog({
|
||||||
id: 'folder-settings',
|
id: 'folder-settings',
|
||||||
title: (
|
title: null,
|
||||||
<HStack space={2} alignItems="center">
|
|
||||||
<Icon icon="folder_cog" size="xl" color="secondary" />
|
|
||||||
{resolvedModelName(folder)}
|
|
||||||
</HStack>
|
|
||||||
),
|
|
||||||
size: 'lg',
|
size: 'lg',
|
||||||
className: 'h-[50rem]',
|
className: 'h-[50rem]',
|
||||||
noPadding: true,
|
noPadding: true,
|
||||||
|
|||||||
@@ -317,7 +317,7 @@ function TextArg({
|
|||||||
autocompleteFunctions,
|
autocompleteFunctions,
|
||||||
autocompleteVariables,
|
autocompleteVariables,
|
||||||
};
|
};
|
||||||
if (autocompleteVariables || autocompleteFunctions) {
|
if (autocompleteVariables || autocompleteFunctions || arg.completionOptions) {
|
||||||
return <Input {...props} />;
|
return <Input {...props} />;
|
||||||
}
|
}
|
||||||
return <PlainInput {...props} />;
|
return <PlainInput {...props} />;
|
||||||
@@ -360,8 +360,9 @@ function EditorArg({
|
|||||||
className={classNames(
|
className={classNames(
|
||||||
'border border-border rounded-md overflow-hidden px-2 py-1',
|
'border border-border rounded-md overflow-hidden px-2 py-1',
|
||||||
'focus-within:border-border-focus',
|
'focus-within:border-border-focus',
|
||||||
'max-h-[10rem]', // So it doesn't take up too much space
|
!arg.rows && 'max-h-[10rem]', // So it doesn't take up too much space
|
||||||
)}
|
)}
|
||||||
|
style={arg.rows ? { height: `${arg.rows * 1.4 + 0.75}rem` } : undefined}
|
||||||
>
|
>
|
||||||
<Editor
|
<Editor
|
||||||
id={id}
|
id={id}
|
||||||
|
|||||||
@@ -1,15 +1,25 @@
|
|||||||
import { createWorkspaceModel, foldersAtom, patchModel } from '@yaakapp-internal/models';
|
import {
|
||||||
|
createWorkspaceModel,
|
||||||
|
foldersAtom,
|
||||||
|
patchModel,
|
||||||
|
} from '@yaakapp-internal/models';
|
||||||
import { useAtomValue } from 'jotai';
|
import { useAtomValue } from 'jotai';
|
||||||
import { useMemo } from 'react';
|
import { Fragment, useMemo } from 'react';
|
||||||
import { useAuthTab } from '../hooks/useAuthTab';
|
import { useAuthTab } from '../hooks/useAuthTab';
|
||||||
import { useEnvironmentsBreakdown } from '../hooks/useEnvironmentsBreakdown';
|
import { useEnvironmentsBreakdown } from '../hooks/useEnvironmentsBreakdown';
|
||||||
import { useHeadersTab } from '../hooks/useHeadersTab';
|
import { useHeadersTab } from '../hooks/useHeadersTab';
|
||||||
import { useInheritedHeaders } from '../hooks/useInheritedHeaders';
|
import { useInheritedHeaders } from '../hooks/useInheritedHeaders';
|
||||||
|
import { useModelAncestors } from '../hooks/useModelAncestors';
|
||||||
|
import { deleteModelWithConfirm } from '../lib/deleteModelWithConfirm';
|
||||||
|
import { hideDialog } from '../lib/dialog';
|
||||||
|
import { CopyIconButton } from './CopyIconButton';
|
||||||
import { Button } from './core/Button';
|
import { Button } from './core/Button';
|
||||||
import { CountBadge } from './core/CountBadge';
|
import { CountBadge } from './core/CountBadge';
|
||||||
|
import { Icon } from './core/Icon';
|
||||||
|
import { InlineCode } from './core/InlineCode';
|
||||||
import { Input } from './core/Input';
|
import { Input } from './core/Input';
|
||||||
import { Link } from './core/Link';
|
import { Link } from './core/Link';
|
||||||
import { VStack } from './core/Stacks';
|
import { HStack, VStack } from './core/Stacks';
|
||||||
import type { TabItem } from './core/Tabs/Tabs';
|
import type { TabItem } from './core/Tabs/Tabs';
|
||||||
import { TabContent, Tabs } from './core/Tabs/Tabs';
|
import { TabContent, Tabs } from './core/Tabs/Tabs';
|
||||||
import { EmptyStateText } from './EmptyStateText';
|
import { EmptyStateText } from './EmptyStateText';
|
||||||
@@ -37,6 +47,8 @@ export type FolderSettingsTab =
|
|||||||
export function FolderSettingsDialog({ folderId, tab }: Props) {
|
export function FolderSettingsDialog({ folderId, tab }: Props) {
|
||||||
const folders = useAtomValue(foldersAtom);
|
const folders = useAtomValue(foldersAtom);
|
||||||
const folder = folders.find((f) => f.id === folderId) ?? null;
|
const folder = folders.find((f) => f.id === folderId) ?? null;
|
||||||
|
const ancestors = useModelAncestors(folder);
|
||||||
|
const breadcrumbs = useMemo(() => ancestors.toReversed(), [ancestors]);
|
||||||
const authTab = useAuthTab(TAB_AUTH, folder);
|
const authTab = useAuthTab(TAB_AUTH, folder);
|
||||||
const headersTab = useHeadersTab(TAB_HEADERS, folder);
|
const headersTab = useHeadersTab(TAB_HEADERS, folder);
|
||||||
const inheritedHeaders = useInheritedHeaders(folder);
|
const inheritedHeaders = useInheritedHeaders(folder);
|
||||||
@@ -67,76 +79,132 @@ export function FolderSettingsDialog({ folderId, tab }: Props) {
|
|||||||
if (folder == null) return null;
|
if (folder == null) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tabs
|
<div className="h-full flex flex-col">
|
||||||
defaultValue={tab ?? TAB_GENERAL}
|
<div className="flex items-center gap-3 px-6 pr-10 mt-4 mb-2 min-w-0 text-xl">
|
||||||
label="Folder Settings"
|
<Icon icon="folder_cog" size="lg" color="secondary" className="flex-shrink-0" />
|
||||||
className="pt-2 pb-2 pl-3 pr-1"
|
<div className="flex items-center gap-1.5 font-semibold text-text min-w-0 overflow-hidden flex-1">
|
||||||
layout="horizontal"
|
{breadcrumbs.map((item, index) => (
|
||||||
addBorders
|
<Fragment key={item.id}>
|
||||||
tabs={tabs}
|
{index > 0 && (
|
||||||
>
|
<Icon
|
||||||
<TabContent value={TAB_AUTH} className="overflow-y-auto h-full px-4">
|
icon="chevron_right"
|
||||||
<HttpAuthenticationEditor model={folder} />
|
size="lg"
|
||||||
</TabContent>
|
className="opacity-50 flex-shrink-0"
|
||||||
<TabContent value={TAB_GENERAL} className="overflow-y-auto h-full px-4">
|
/>
|
||||||
<VStack space={3} className="pb-3 h-full">
|
)}
|
||||||
<Input
|
<span className="text-text-subtle truncate min-w-0" title={item.name}>
|
||||||
label="Folder Name"
|
{item.name}
|
||||||
defaultValue={folder.name}
|
</span>
|
||||||
onChange={(name) => patchModel(folder, { name })}
|
</Fragment>
|
||||||
stateKey={`name.${folder.id}`}
|
))}
|
||||||
/>
|
{breadcrumbs.length > 0 && (
|
||||||
<MarkdownEditor
|
<Icon icon="chevron_right" size="lg" className="opacity-50 flex-shrink-0" />
|
||||||
name="folder-description"
|
)}
|
||||||
placeholder="Folder description"
|
<span
|
||||||
className="border border-border px-2"
|
className="whitespace-nowrap"
|
||||||
defaultValue={folder.description}
|
title={folder.name}
|
||||||
stateKey={`description.${folder.id}`}
|
>
|
||||||
onChange={(description) => patchModel(folder, { description })}
|
{folder.name}
|
||||||
/>
|
</span>
|
||||||
</VStack>
|
</div>
|
||||||
</TabContent>
|
</div>
|
||||||
<TabContent value={TAB_HEADERS} className="overflow-y-auto h-full px-4">
|
|
||||||
<HeadersEditor
|
<Tabs
|
||||||
inheritedHeaders={inheritedHeaders}
|
defaultValue={tab ?? TAB_GENERAL}
|
||||||
forceUpdateKey={folder.id}
|
label="Folder Settings"
|
||||||
headers={folder.headers}
|
className="pt-2 pb-2 pl-3 pr-1 flex-1"
|
||||||
onChange={(headers) => patchModel(folder, { headers })}
|
layout="horizontal"
|
||||||
stateKey={`headers.${folder.id}`}
|
addBorders
|
||||||
/>
|
tabs={tabs}
|
||||||
</TabContent>
|
>
|
||||||
<TabContent value={TAB_VARIABLES} className="overflow-y-auto h-full px-4">
|
<TabContent value={TAB_AUTH} className="overflow-y-auto h-full px-4">
|
||||||
{folderEnvironment == null ? (
|
<HttpAuthenticationEditor model={folder} />
|
||||||
<EmptyStateText>
|
</TabContent>
|
||||||
<VStack alignItems="center" space={1.5}>
|
<TabContent value={TAB_GENERAL} className="overflow-y-auto h-full px-4">
|
||||||
<p>
|
<div className="grid grid-rows-[auto_minmax(0,1fr)_auto] gap-3 pb-3 h-full">
|
||||||
Override{' '}
|
<Input
|
||||||
<Link href="https://yaak.app/docs/using-yaak/environments-and-variables">
|
label="Folder Name"
|
||||||
Variables
|
defaultValue={folder.name}
|
||||||
</Link>{' '}
|
onChange={(name) => patchModel(folder, { name })}
|
||||||
for requests within this folder.
|
stateKey={`name.${folder.id}`}
|
||||||
</p>
|
/>
|
||||||
|
<MarkdownEditor
|
||||||
|
name="folder-description"
|
||||||
|
placeholder="Folder description"
|
||||||
|
className="border border-border px-2"
|
||||||
|
defaultValue={folder.description}
|
||||||
|
stateKey={`description.${folder.id}`}
|
||||||
|
onChange={(description) => patchModel(folder, { description })}
|
||||||
|
/>
|
||||||
|
<HStack alignItems="center" justifyContent="between" className="w-full">
|
||||||
<Button
|
<Button
|
||||||
variant="border"
|
|
||||||
size="sm"
|
|
||||||
onClick={async () => {
|
onClick={async () => {
|
||||||
await createWorkspaceModel({
|
const didDelete = await deleteModelWithConfirm(folder);
|
||||||
workspaceId: folder.workspaceId,
|
if (didDelete) {
|
||||||
parentModel: 'folder',
|
hideDialog('folder-settings');
|
||||||
parentId: folder.id,
|
}
|
||||||
model: 'environment',
|
|
||||||
name: 'Folder Environment',
|
|
||||||
});
|
|
||||||
}}
|
}}
|
||||||
|
color="danger"
|
||||||
|
variant="border"
|
||||||
|
size="xs"
|
||||||
>
|
>
|
||||||
Create Folder Environment
|
Delete Folder
|
||||||
</Button>
|
</Button>
|
||||||
</VStack>
|
<InlineCode className="flex gap-1 items-center text-primary pl-2.5">
|
||||||
</EmptyStateText>
|
{folder.id}
|
||||||
) : (
|
<CopyIconButton
|
||||||
<EnvironmentEditor hideName environment={folderEnvironment} />
|
className="opacity-70 !text-primary"
|
||||||
)}
|
size="2xs"
|
||||||
</TabContent>
|
iconSize="sm"
|
||||||
</Tabs>
|
title="Copy folder ID"
|
||||||
|
text={folder.id}
|
||||||
|
/>
|
||||||
|
</InlineCode>
|
||||||
|
</HStack>
|
||||||
|
</div>
|
||||||
|
</TabContent>
|
||||||
|
<TabContent value={TAB_HEADERS} className="overflow-y-auto h-full px-4">
|
||||||
|
<HeadersEditor
|
||||||
|
inheritedHeaders={inheritedHeaders}
|
||||||
|
forceUpdateKey={folder.id}
|
||||||
|
headers={folder.headers}
|
||||||
|
onChange={(headers) => patchModel(folder, { headers })}
|
||||||
|
stateKey={`headers.${folder.id}`}
|
||||||
|
/>
|
||||||
|
</TabContent>
|
||||||
|
<TabContent value={TAB_VARIABLES} className="overflow-y-auto h-full px-4">
|
||||||
|
{folderEnvironment == null ? (
|
||||||
|
<EmptyStateText>
|
||||||
|
<VStack alignItems="center" space={1.5}>
|
||||||
|
<p>
|
||||||
|
Override{' '}
|
||||||
|
<Link href="https://yaak.app/docs/using-yaak/environments-and-variables">
|
||||||
|
Variables
|
||||||
|
</Link>{' '}
|
||||||
|
for requests within this folder.
|
||||||
|
</p>
|
||||||
|
<Button
|
||||||
|
variant="border"
|
||||||
|
size="sm"
|
||||||
|
onClick={async () => {
|
||||||
|
await createWorkspaceModel({
|
||||||
|
workspaceId: folder.workspaceId,
|
||||||
|
parentModel: 'folder',
|
||||||
|
parentId: folder.id,
|
||||||
|
model: 'environment',
|
||||||
|
name: 'Folder Environment',
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Create Folder Environment
|
||||||
|
</Button>
|
||||||
|
</VStack>
|
||||||
|
</EmptyStateText>
|
||||||
|
) : (
|
||||||
|
<EnvironmentEditor hideName environment={folderEnvironment} />
|
||||||
|
)}
|
||||||
|
</TabContent>
|
||||||
|
</Tabs>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -98,13 +98,14 @@ export function GrpcResponsePane({ style, methodType, activeRequest }: Props) {
|
|||||||
renderRow={({ event, isActive, onClick }) => (
|
renderRow={({ event, isActive, onClick }) => (
|
||||||
<GrpcEventRow event={event} isActive={isActive} onClick={onClick} />
|
<GrpcEventRow event={event} isActive={isActive} onClick={onClick} />
|
||||||
)}
|
)}
|
||||||
renderDetail={({ event }) => (
|
renderDetail={({ event, onClose }) => (
|
||||||
<GrpcEventDetail
|
<GrpcEventDetail
|
||||||
event={event}
|
event={event}
|
||||||
showLarge={showLarge}
|
showLarge={showLarge}
|
||||||
showingLarge={showingLarge}
|
showingLarge={showingLarge}
|
||||||
setShowLarge={setShowLarge}
|
setShowLarge={setShowLarge}
|
||||||
setShowingLarge={setShowingLarge}
|
setShowingLarge={setShowingLarge}
|
||||||
|
onClose={onClose}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
@@ -147,19 +148,26 @@ function GrpcEventDetail({
|
|||||||
showingLarge,
|
showingLarge,
|
||||||
setShowLarge,
|
setShowLarge,
|
||||||
setShowingLarge,
|
setShowingLarge,
|
||||||
|
onClose,
|
||||||
}: {
|
}: {
|
||||||
event: GrpcEvent;
|
event: GrpcEvent;
|
||||||
showLarge: boolean;
|
showLarge: boolean;
|
||||||
showingLarge: boolean;
|
showingLarge: boolean;
|
||||||
setShowLarge: (v: boolean) => void;
|
setShowLarge: (v: boolean) => void;
|
||||||
setShowingLarge: (v: boolean) => void;
|
setShowingLarge: (v: boolean) => void;
|
||||||
|
onClose: () => void;
|
||||||
}) {
|
}) {
|
||||||
if (event.eventType === 'client_message' || event.eventType === 'server_message') {
|
if (event.eventType === 'client_message' || event.eventType === 'server_message') {
|
||||||
const title = `Message ${event.eventType === 'client_message' ? 'Sent' : 'Received'}`;
|
const title = `Message ${event.eventType === 'client_message' ? 'Sent' : 'Received'}`;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="h-full grid grid-rows-[auto_minmax(0,1fr)]">
|
<div className="h-full grid grid-rows-[auto_minmax(0,1fr)]">
|
||||||
<EventDetailHeader title={title} timestamp={event.createdAt} copyText={event.content} />
|
<EventDetailHeader
|
||||||
|
title={title}
|
||||||
|
timestamp={event.createdAt}
|
||||||
|
copyText={event.content}
|
||||||
|
onClose={onClose}
|
||||||
|
/>
|
||||||
{!showLarge && event.content.length > 1000 * 1000 ? (
|
{!showLarge && event.content.length > 1000 * 1000 ? (
|
||||||
<VStack space={2} className="italic text-text-subtlest">
|
<VStack space={2} className="italic text-text-subtlest">
|
||||||
Message previews larger than 1MB are hidden
|
Message previews larger than 1MB are hidden
|
||||||
@@ -197,7 +205,7 @@ function GrpcEventDetail({
|
|||||||
// Error or connection_end - show metadata/trailers
|
// Error or connection_end - show metadata/trailers
|
||||||
return (
|
return (
|
||||||
<div className="h-full grid grid-rows-[auto_minmax(0,1fr)]">
|
<div className="h-full grid grid-rows-[auto_minmax(0,1fr)]">
|
||||||
<EventDetailHeader title={event.content} timestamp={event.createdAt} />
|
<EventDetailHeader title={event.content} timestamp={event.createdAt} onClose={onClose} />
|
||||||
{event.error && (
|
{event.error && (
|
||||||
<div className="select-text cursor-text text-sm font-mono py-1 text-warning">
|
<div className="select-text cursor-text text-sm font-mono py-1 text-warning">
|
||||||
{event.error}
|
{event.error}
|
||||||
|
|||||||
@@ -239,7 +239,7 @@ export function HttpResponsePane({ style, className, activeRequestId }: Props) {
|
|||||||
<HttpMultipartViewer response={activeResponse} />
|
<HttpMultipartViewer response={activeResponse} />
|
||||||
) : mimeType?.match(/pdf/i) ? (
|
) : mimeType?.match(/pdf/i) ? (
|
||||||
<EnsureCompleteResponse response={activeResponse} Component={PdfViewer} />
|
<EnsureCompleteResponse response={activeResponse} Component={PdfViewer} />
|
||||||
) : mimeType?.match(/csv|tab-separated/i) ? (
|
) : mimeType?.match(/csv|tab-separated/i) && viewMode === 'pretty' ? (
|
||||||
<HttpCsvViewer className="pb-2" response={activeResponse} />
|
<HttpCsvViewer className="pb-2" response={activeResponse} />
|
||||||
) : (
|
) : (
|
||||||
<HTMLOrTextViewer
|
<HTMLOrTextViewer
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import { useStateWithDeps } from '../hooks/useStateWithDeps';
|
|||||||
import { languageFromContentType } from '../lib/contentType';
|
import { languageFromContentType } from '../lib/contentType';
|
||||||
import { Button } from './core/Button';
|
import { Button } from './core/Button';
|
||||||
import { Editor } from './core/Editor/LazyEditor';
|
import { Editor } from './core/Editor/LazyEditor';
|
||||||
import { EventDetailHeader, EventViewer, type EventDetailAction } from './core/EventViewer';
|
import { type EventDetailAction, EventDetailHeader, EventViewer } from './core/EventViewer';
|
||||||
import { EventViewerRow } from './core/EventViewerRow';
|
import { EventViewerRow } from './core/EventViewerRow';
|
||||||
import { HotkeyList } from './core/HotkeyList';
|
import { HotkeyList } from './core/HotkeyList';
|
||||||
import { Icon } from './core/Icon';
|
import { Icon } from './core/Icon';
|
||||||
@@ -75,7 +75,7 @@ export function WebsocketResponsePane({ activeRequest }: Props) {
|
|||||||
renderRow={({ event, isActive, onClick }) => (
|
renderRow={({ event, isActive, onClick }) => (
|
||||||
<WebsocketEventRow event={event} isActive={isActive} onClick={onClick} />
|
<WebsocketEventRow event={event} isActive={isActive} onClick={onClick} />
|
||||||
)}
|
)}
|
||||||
renderDetail={({ event, index }) => (
|
renderDetail={({ event, index, onClose }) => (
|
||||||
<WebsocketEventDetail
|
<WebsocketEventDetail
|
||||||
event={event}
|
event={event}
|
||||||
hexDump={hexDumps[index] ?? event.messageType === 'binary'}
|
hexDump={hexDumps[index] ?? event.messageType === 'binary'}
|
||||||
@@ -84,6 +84,7 @@ export function WebsocketResponsePane({ activeRequest }: Props) {
|
|||||||
showingLarge={showingLarge}
|
showingLarge={showingLarge}
|
||||||
setShowLarge={setShowLarge}
|
setShowLarge={setShowLarge}
|
||||||
setShowingLarge={setShowingLarge}
|
setShowingLarge={setShowingLarge}
|
||||||
|
onClose={onClose}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
@@ -145,6 +146,7 @@ function WebsocketEventDetail({
|
|||||||
showingLarge,
|
showingLarge,
|
||||||
setShowLarge,
|
setShowLarge,
|
||||||
setShowingLarge,
|
setShowingLarge,
|
||||||
|
onClose,
|
||||||
}: {
|
}: {
|
||||||
event: WebsocketEvent;
|
event: WebsocketEvent;
|
||||||
hexDump: boolean;
|
hexDump: boolean;
|
||||||
@@ -153,6 +155,7 @@ function WebsocketEventDetail({
|
|||||||
showingLarge: boolean;
|
showingLarge: boolean;
|
||||||
setShowLarge: (v: boolean) => void;
|
setShowLarge: (v: boolean) => void;
|
||||||
setShowingLarge: (v: boolean) => void;
|
setShowingLarge: (v: boolean) => void;
|
||||||
|
onClose: () => void;
|
||||||
}) {
|
}) {
|
||||||
const message = useMemo(() => {
|
const message = useMemo(() => {
|
||||||
if (hexDump) {
|
if (hexDump) {
|
||||||
@@ -185,11 +188,12 @@ function WebsocketEventDetail({
|
|||||||
return (
|
return (
|
||||||
<div className="h-full grid grid-rows-[auto_minmax(0,1fr)]">
|
<div className="h-full grid grid-rows-[auto_minmax(0,1fr)]">
|
||||||
<EventDetailHeader
|
<EventDetailHeader
|
||||||
title={title}
|
title={title}
|
||||||
timestamp={event.createdAt}
|
timestamp={event.createdAt}
|
||||||
actions={actions}
|
actions={actions}
|
||||||
copyText={formattedMessage || undefined}
|
copyText={formattedMessage || undefined}
|
||||||
/>
|
onClose={onClose}
|
||||||
|
/>
|
||||||
{!showLarge && event.message.length > 1000 * 1000 ? (
|
{!showLarge && event.message.length > 1000 * 1000 ? (
|
||||||
<VStack space={2} className="italic text-text-subtlest">
|
<VStack space={2} className="italic text-text-subtlest">
|
||||||
Message previews larger than 1MB are hidden
|
Message previews larger than 1MB are hidden
|
||||||
|
|||||||
@@ -135,6 +135,7 @@ export function Workspace() {
|
|||||||
open={!floatingSidebarHidden}
|
open={!floatingSidebarHidden}
|
||||||
portalName="sidebar"
|
portalName="sidebar"
|
||||||
onClose={() => setFloatingSidebarHidden(true)}
|
onClose={() => setFloatingSidebarHidden(true)}
|
||||||
|
zIndex={20}
|
||||||
>
|
>
|
||||||
<m.div
|
<m.div
|
||||||
initial={{ opacity: 0, x: -20 }}
|
initial={{ opacity: 0, x: -20 }}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { useMemo } from 'react';
|
|||||||
import { Overlay } from '../Overlay';
|
import { Overlay } from '../Overlay';
|
||||||
import { Heading } from './Heading';
|
import { Heading } from './Heading';
|
||||||
import { IconButton } from './IconButton';
|
import { IconButton } from './IconButton';
|
||||||
|
import { DialogSize } from '@yaakapp-internal/plugins';
|
||||||
|
|
||||||
export interface DialogProps {
|
export interface DialogProps {
|
||||||
children: ReactNode;
|
children: ReactNode;
|
||||||
@@ -14,7 +15,7 @@ export interface DialogProps {
|
|||||||
title?: ReactNode;
|
title?: ReactNode;
|
||||||
description?: ReactNode;
|
description?: ReactNode;
|
||||||
className?: string;
|
className?: string;
|
||||||
size?: 'sm' | 'md' | 'lg' | 'full' | 'dynamic';
|
size?: DialogSize;
|
||||||
hideX?: boolean;
|
hideX?: boolean;
|
||||||
noPadding?: boolean;
|
noPadding?: boolean;
|
||||||
noScroll?: boolean;
|
noScroll?: boolean;
|
||||||
|
|||||||
@@ -5,11 +5,14 @@ import {
|
|||||||
completionKeymap,
|
completionKeymap,
|
||||||
} from '@codemirror/autocomplete';
|
} from '@codemirror/autocomplete';
|
||||||
import { history, historyKeymap } from '@codemirror/commands';
|
import { history, historyKeymap } from '@codemirror/commands';
|
||||||
|
import { go } from '@codemirror/lang-go';
|
||||||
|
import { java } from '@codemirror/lang-java';
|
||||||
import { javascript } from '@codemirror/lang-javascript';
|
import { javascript } from '@codemirror/lang-javascript';
|
||||||
import { json } from '@codemirror/lang-json';
|
import { json } from '@codemirror/lang-json';
|
||||||
import { markdown } from '@codemirror/lang-markdown';
|
import { markdown } from '@codemirror/lang-markdown';
|
||||||
|
import { php } from '@codemirror/lang-php';
|
||||||
|
import { python } from '@codemirror/lang-python';
|
||||||
import { xml } from '@codemirror/lang-xml';
|
import { xml } from '@codemirror/lang-xml';
|
||||||
import type { LanguageSupport } from '@codemirror/language';
|
|
||||||
import {
|
import {
|
||||||
bracketMatching,
|
bracketMatching,
|
||||||
codeFolding,
|
codeFolding,
|
||||||
@@ -17,8 +20,19 @@ import {
|
|||||||
foldKeymap,
|
foldKeymap,
|
||||||
HighlightStyle,
|
HighlightStyle,
|
||||||
indentOnInput,
|
indentOnInput,
|
||||||
|
LanguageSupport,
|
||||||
|
StreamLanguage,
|
||||||
syntaxHighlighting,
|
syntaxHighlighting,
|
||||||
} from '@codemirror/language';
|
} from '@codemirror/language';
|
||||||
|
import { c, csharp, kotlin, objectiveC } from '@codemirror/legacy-modes/mode/clike';
|
||||||
|
import { clojure } from '@codemirror/legacy-modes/mode/clojure';
|
||||||
|
import { http } from '@codemirror/legacy-modes/mode/http';
|
||||||
|
import { oCaml } from '@codemirror/legacy-modes/mode/mllike';
|
||||||
|
import { powerShell } from '@codemirror/legacy-modes/mode/powershell';
|
||||||
|
import { r } from '@codemirror/legacy-modes/mode/r';
|
||||||
|
import { ruby } from '@codemirror/legacy-modes/mode/ruby';
|
||||||
|
import { shell } from '@codemirror/legacy-modes/mode/shell';
|
||||||
|
import { swift } from '@codemirror/legacy-modes/mode/swift';
|
||||||
import { linter, lintGutter, lintKeymap } from '@codemirror/lint';
|
import { linter, lintGutter, lintKeymap } from '@codemirror/lint';
|
||||||
|
|
||||||
import { search, searchKeymap } from '@codemirror/search';
|
import { search, searchKeymap } from '@codemirror/search';
|
||||||
@@ -83,6 +97,10 @@ const syntaxTheme = EditorView.theme({}, { dark: true });
|
|||||||
|
|
||||||
const closeBracketsExtensions: Extension = [closeBrackets(), keymap.of([...closeBracketsKeymap])];
|
const closeBracketsExtensions: Extension = [closeBrackets(), keymap.of([...closeBracketsKeymap])];
|
||||||
|
|
||||||
|
const legacyLang = (mode: Parameters<typeof StreamLanguage.define>[0]) => {
|
||||||
|
return () => new LanguageSupport(StreamLanguage.define(mode));
|
||||||
|
};
|
||||||
|
|
||||||
const syntaxExtensions: Record<
|
const syntaxExtensions: Record<
|
||||||
NonNullable<EditorProps['language']>,
|
NonNullable<EditorProps['language']>,
|
||||||
null | (() => LanguageSupport)
|
null | (() => LanguageSupport)
|
||||||
@@ -98,6 +116,22 @@ const syntaxExtensions: Record<
|
|||||||
text: text,
|
text: text,
|
||||||
timeline: timeline,
|
timeline: timeline,
|
||||||
markdown: markdown,
|
markdown: markdown,
|
||||||
|
c: legacyLang(c),
|
||||||
|
clojure: legacyLang(clojure),
|
||||||
|
csharp: legacyLang(csharp),
|
||||||
|
go: go,
|
||||||
|
http: legacyLang(http),
|
||||||
|
java: java,
|
||||||
|
kotlin: legacyLang(kotlin),
|
||||||
|
objective_c: legacyLang(objectiveC),
|
||||||
|
ocaml: legacyLang(oCaml),
|
||||||
|
php: php,
|
||||||
|
powershell: legacyLang(powerShell),
|
||||||
|
python: python,
|
||||||
|
r: legacyLang(r),
|
||||||
|
ruby: legacyLang(ruby),
|
||||||
|
shell: legacyLang(shell),
|
||||||
|
swift: legacyLang(swift),
|
||||||
};
|
};
|
||||||
|
|
||||||
const closeBracketsFor: (keyof typeof syntaxExtensions)[] = ['json', 'javascript', 'graphql'];
|
const closeBracketsFor: (keyof typeof syntaxExtensions)[] = ['json', 'javascript', 'graphql'];
|
||||||
|
|||||||
@@ -111,6 +111,7 @@ import {
|
|||||||
RefreshCcwIcon,
|
RefreshCcwIcon,
|
||||||
RefreshCwIcon,
|
RefreshCwIcon,
|
||||||
RocketIcon,
|
RocketIcon,
|
||||||
|
RotateCcwIcon,
|
||||||
Rows2Icon,
|
Rows2Icon,
|
||||||
SaveIcon,
|
SaveIcon,
|
||||||
SearchIcon,
|
SearchIcon,
|
||||||
@@ -249,6 +250,7 @@ const icons = {
|
|||||||
puzzle: PuzzleIcon,
|
puzzle: PuzzleIcon,
|
||||||
refresh: RefreshCwIcon,
|
refresh: RefreshCwIcon,
|
||||||
rocket: RocketIcon,
|
rocket: RocketIcon,
|
||||||
|
rotate_ccw: RotateCcwIcon,
|
||||||
rows_2: Rows2Icon,
|
rows_2: Rows2Icon,
|
||||||
save: SaveIcon,
|
save: SaveIcon,
|
||||||
search: SearchIcon,
|
search: SearchIcon,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import type { FormInput, JsonPrimitive } from '@yaakapp-internal/plugins';
|
import type { FormInput, JsonPrimitive } from '@yaakapp-internal/plugins';
|
||||||
import type { FormEvent } from 'react';
|
import type { FormEvent } from 'react';
|
||||||
import { useCallback, useRef, useState } from 'react';
|
import { useCallback, useEffect, useRef, useState } from 'react';
|
||||||
import { generateId } from '../../lib/generateId';
|
import { generateId } from '../../lib/generateId';
|
||||||
import { DynamicForm } from '../DynamicForm';
|
import { DynamicForm } from '../DynamicForm';
|
||||||
import { Button } from './Button';
|
import { Button } from './Button';
|
||||||
@@ -12,16 +12,21 @@ export interface PromptProps {
|
|||||||
onResult: (value: Record<string, JsonPrimitive> | null) => void;
|
onResult: (value: Record<string, JsonPrimitive> | null) => void;
|
||||||
confirmText?: string;
|
confirmText?: string;
|
||||||
cancelText?: string;
|
cancelText?: string;
|
||||||
|
onValuesChange?: (values: Record<string, JsonPrimitive>) => void;
|
||||||
|
onInputsUpdated?: (cb: (inputs: FormInput[]) => void) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function Prompt({
|
export function Prompt({
|
||||||
onCancel,
|
onCancel,
|
||||||
inputs,
|
inputs: initialInputs,
|
||||||
onResult,
|
onResult,
|
||||||
confirmText = 'Confirm',
|
confirmText = 'Confirm',
|
||||||
cancelText = 'Cancel',
|
cancelText = 'Cancel',
|
||||||
|
onValuesChange,
|
||||||
|
onInputsUpdated,
|
||||||
}: PromptProps) {
|
}: PromptProps) {
|
||||||
const [value, setValue] = useState<Record<string, JsonPrimitive>>({});
|
const [value, setValue] = useState<Record<string, JsonPrimitive>>({});
|
||||||
|
const [inputs, setInputs] = useState<FormInput[]>(initialInputs);
|
||||||
const handleSubmit = useCallback(
|
const handleSubmit = useCallback(
|
||||||
(e: FormEvent<HTMLFormElement>) => {
|
(e: FormEvent<HTMLFormElement>) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@@ -30,6 +35,16 @@ export function Prompt({
|
|||||||
[onResult, value],
|
[onResult, value],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Register callback for external input updates (from plugin dynamic resolution)
|
||||||
|
useEffect(() => {
|
||||||
|
onInputsUpdated?.(setInputs);
|
||||||
|
}, [onInputsUpdated]);
|
||||||
|
|
||||||
|
// Notify of value changes for dynamic resolution
|
||||||
|
useEffect(() => {
|
||||||
|
onValuesChange?.(value);
|
||||||
|
}, [value, onValuesChange]);
|
||||||
|
|
||||||
const id = `prompt.form.${useRef(generateId()).current}`;
|
const id = `prompt.form.${useRef(generateId()).current}`;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
66
src-web/components/core/RadioCards.tsx
Normal file
66
src-web/components/core/RadioCards.tsx
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
import classNames from 'classnames';
|
||||||
|
import type { ReactNode } from 'react';
|
||||||
|
|
||||||
|
export interface RadioCardOption<T extends string> {
|
||||||
|
value: T;
|
||||||
|
label: ReactNode;
|
||||||
|
description?: ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RadioCardsProps<T extends string> {
|
||||||
|
value: T | null;
|
||||||
|
onChange: (value: T) => void;
|
||||||
|
options: RadioCardOption<T>[];
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function RadioCards<T extends string>({
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
options,
|
||||||
|
name,
|
||||||
|
}: RadioCardsProps<T>) {
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
{options.map((option) => {
|
||||||
|
const selected = value === option.value;
|
||||||
|
return (
|
||||||
|
<label
|
||||||
|
key={option.value}
|
||||||
|
className={classNames(
|
||||||
|
'flex items-start gap-3 p-3 rounded-lg border cursor-pointer',
|
||||||
|
'transition-colors',
|
||||||
|
selected
|
||||||
|
? 'border-border-focus'
|
||||||
|
: 'border-border-subtle hocus:border-text-subtlest',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
type="radio"
|
||||||
|
name={name}
|
||||||
|
value={option.value}
|
||||||
|
checked={selected}
|
||||||
|
onChange={() => onChange(option.value)}
|
||||||
|
className="sr-only"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className={classNames(
|
||||||
|
'mt-1 w-4 h-4 flex-shrink-0 rounded-full border',
|
||||||
|
'flex items-center justify-center',
|
||||||
|
selected ? 'border-focus' : 'border-border',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{selected && <div className="w-2 h-2 rounded-full bg-text" />}
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-0.5">
|
||||||
|
<span className="font-semibold text-text">{option.label}</span>
|
||||||
|
{option.description && (
|
||||||
|
<span className="text-sm text-text-subtle">{option.description}</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -20,7 +20,7 @@ import { InlineCode } from '../core/InlineCode';
|
|||||||
import { gitCallbacks } from './callbacks';
|
import { gitCallbacks } from './callbacks';
|
||||||
import { GitCommitDialog } from './GitCommitDialog';
|
import { GitCommitDialog } from './GitCommitDialog';
|
||||||
import { GitRemotesDialog } from './GitRemotesDialog';
|
import { GitRemotesDialog } from './GitRemotesDialog';
|
||||||
import { handlePullResult } from './git-util';
|
import { handlePullResult, handlePushResult } from './git-util';
|
||||||
import { HistoryDialog } from './HistoryDialog';
|
import { HistoryDialog } from './HistoryDialog';
|
||||||
|
|
||||||
export function GitDropdown() {
|
export function GitDropdown() {
|
||||||
@@ -48,6 +48,7 @@ function SyncDropdownWithSyncDir({ syncDir }: { syncDir: string }) {
|
|||||||
push,
|
push,
|
||||||
pull,
|
pull,
|
||||||
checkout,
|
checkout,
|
||||||
|
resetChanges,
|
||||||
init,
|
init,
|
||||||
},
|
},
|
||||||
] = useGit(syncDir, gitCallbacks(syncDir));
|
] = useGit(syncDir, gitCallbacks(syncDir));
|
||||||
@@ -72,6 +73,9 @@ function SyncDropdownWithSyncDir({ syncDir }: { syncDir: string }) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const currentBranch = status.data.headRefShorthand;
|
const currentBranch = status.data.headRefShorthand;
|
||||||
|
const hasChanges = status.data.entries.some((e) => e.status !== 'current');
|
||||||
|
const hasRemotes = (status.data.origins ?? []).length > 0;
|
||||||
|
const { ahead, behind } = status.data;
|
||||||
|
|
||||||
const tryCheckout = (branch: string, force: boolean) => {
|
const tryCheckout = (branch: string, force: boolean) => {
|
||||||
checkout.mutate(
|
checkout.mutate(
|
||||||
@@ -168,12 +172,13 @@ function SyncDropdownWithSyncDir({ syncDir }: { syncDir: string }) {
|
|||||||
{ type: 'separator' },
|
{ type: 'separator' },
|
||||||
{
|
{
|
||||||
label: 'Push',
|
label: 'Push',
|
||||||
|
disabled: !hasRemotes || ahead === 0,
|
||||||
leftSlot: <Icon icon="arrow_up_from_line" />,
|
leftSlot: <Icon icon="arrow_up_from_line" />,
|
||||||
waitForOnSelect: true,
|
waitForOnSelect: true,
|
||||||
async onSelect() {
|
async onSelect() {
|
||||||
await push.mutateAsync(undefined, {
|
await push.mutateAsync(undefined, {
|
||||||
disableToastError: true,
|
disableToastError: true,
|
||||||
onSuccess: handlePullResult,
|
onSuccess: handlePushResult,
|
||||||
onError(err) {
|
onError(err) {
|
||||||
showErrorToast({
|
showErrorToast({
|
||||||
id: 'git-push-error',
|
id: 'git-push-error',
|
||||||
@@ -186,7 +191,7 @@ function SyncDropdownWithSyncDir({ syncDir }: { syncDir: string }) {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Pull',
|
label: 'Pull',
|
||||||
hidden: (status.data?.origins ?? []).length === 0,
|
disabled: !hasRemotes || behind === 0,
|
||||||
leftSlot: <Icon icon="arrow_down_to_line" />,
|
leftSlot: <Icon icon="arrow_down_to_line" />,
|
||||||
waitForOnSelect: true,
|
waitForOnSelect: true,
|
||||||
async onSelect() {
|
async onSelect() {
|
||||||
@@ -205,6 +210,7 @@ function SyncDropdownWithSyncDir({ syncDir }: { syncDir: string }) {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Commit...',
|
label: 'Commit...',
|
||||||
|
disabled: !hasChanges,
|
||||||
leftSlot: <Icon icon="git_commit_vertical" />,
|
leftSlot: <Icon icon="git_commit_vertical" />,
|
||||||
onSelect() {
|
onSelect() {
|
||||||
showDialog({
|
showDialog({
|
||||||
@@ -218,6 +224,41 @@ function SyncDropdownWithSyncDir({ syncDir }: { syncDir: string }) {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: 'Reset Changes',
|
||||||
|
hidden: !hasChanges,
|
||||||
|
leftSlot: <Icon icon="rotate_ccw" />,
|
||||||
|
color: 'danger',
|
||||||
|
async onSelect() {
|
||||||
|
const confirmed = await showConfirm({
|
||||||
|
id: 'git-reset-changes',
|
||||||
|
title: 'Reset Changes',
|
||||||
|
description: 'This will discard all uncommitted changes. This cannot be undone.',
|
||||||
|
confirmText: 'Reset',
|
||||||
|
color: 'danger',
|
||||||
|
});
|
||||||
|
if (!confirmed) return;
|
||||||
|
|
||||||
|
await resetChanges.mutateAsync(undefined, {
|
||||||
|
disableToastError: true,
|
||||||
|
onSuccess() {
|
||||||
|
showToast({
|
||||||
|
id: 'git-reset-success',
|
||||||
|
message: 'Changes have been reset',
|
||||||
|
color: 'success',
|
||||||
|
});
|
||||||
|
sync({ force: true });
|
||||||
|
},
|
||||||
|
onError(err) {
|
||||||
|
showErrorToast({
|
||||||
|
id: 'git-reset-error',
|
||||||
|
title: 'Error resetting changes',
|
||||||
|
message: String(err),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
{ type: 'separator', label: 'Branches', hidden: localBranches.length < 1 },
|
{ type: 'separator', label: 'Branches', hidden: localBranches.length < 1 },
|
||||||
...localBranches.map((branch) => {
|
...localBranches.map((branch) => {
|
||||||
const isCurrent = currentBranch === branch;
|
const isCurrent = currentBranch === branch;
|
||||||
@@ -463,8 +504,14 @@ function SyncDropdownWithSyncDir({ syncDir }: { syncDir: string }) {
|
|||||||
return (
|
return (
|
||||||
<Dropdown fullWidth items={items} onOpen={fetchAll.mutate}>
|
<Dropdown fullWidth items={items} onOpen={fetchAll.mutate}>
|
||||||
<GitMenuButton>
|
<GitMenuButton>
|
||||||
<InlineCode>{currentBranch}</InlineCode>
|
<InlineCode className="flex items-center gap-1">
|
||||||
<Icon icon="git_branch" size="sm" />
|
<Icon icon="git_branch" size="xs" className="opacity-50" />
|
||||||
|
{currentBranch}
|
||||||
|
</InlineCode>
|
||||||
|
<div className="flex items-center gap-1.5">
|
||||||
|
{ahead > 0 && <span className="text-xs flex items-center gap-0.5"><span className="text-primary">↗</span>{ahead}</span>}
|
||||||
|
{behind > 0 && <span className="text-xs flex items-center gap-0.5"><span className="text-info">↙</span>{behind}</span>}
|
||||||
|
</div>
|
||||||
</GitMenuButton>
|
</GitMenuButton>
|
||||||
</Dropdown>
|
</Dropdown>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
import type { GitCallbacks } from '@yaakapp-internal/git';
|
import type { GitCallbacks } from '@yaakapp-internal/git';
|
||||||
|
import { sync } from '../../init/sync';
|
||||||
import { promptCredentials } from './credentials';
|
import { promptCredentials } from './credentials';
|
||||||
|
import { promptDivergedStrategy } from './diverged';
|
||||||
|
import { promptUncommittedChangesStrategy } from './uncommitted';
|
||||||
import { addGitRemote } from './showAddRemoteDialog';
|
import { addGitRemote } from './showAddRemoteDialog';
|
||||||
|
|
||||||
export function gitCallbacks(dir: string): GitCallbacks {
|
export function gitCallbacks(dir: string): GitCallbacks {
|
||||||
@@ -12,5 +15,12 @@ export function gitCallbacks(dir: string): GitCallbacks {
|
|||||||
if (creds == null) throw new Error('Cancelled credentials prompt');
|
if (creds == null) throw new Error('Cancelled credentials prompt');
|
||||||
return creds;
|
return creds;
|
||||||
},
|
},
|
||||||
|
promptDiverged: async ({ remote, branch }) => {
|
||||||
|
return promptDivergedStrategy({ remote, branch });
|
||||||
|
},
|
||||||
|
promptUncommittedChanges: async () => {
|
||||||
|
return promptUncommittedChangesStrategy();
|
||||||
|
},
|
||||||
|
forceSync: () => sync({ force: true }),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
102
src-web/components/git/diverged.tsx
Normal file
102
src-web/components/git/diverged.tsx
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
import type { DivergedStrategy } from '@yaakapp-internal/git';
|
||||||
|
import { useState } from 'react';
|
||||||
|
import { showDialog } from '../../lib/dialog';
|
||||||
|
import { Button } from '../core/Button';
|
||||||
|
import { InlineCode } from '../core/InlineCode';
|
||||||
|
import { RadioCards } from '../core/RadioCards';
|
||||||
|
import { HStack } from '../core/Stacks';
|
||||||
|
|
||||||
|
type Resolution = 'force_reset' | 'merge';
|
||||||
|
|
||||||
|
const resolutionLabel: Record<Resolution, string> = {
|
||||||
|
force_reset: 'Force Pull',
|
||||||
|
merge: 'Merge',
|
||||||
|
};
|
||||||
|
|
||||||
|
interface DivergedDialogProps {
|
||||||
|
remote: string;
|
||||||
|
branch: string;
|
||||||
|
onResult: (strategy: DivergedStrategy) => void;
|
||||||
|
onHide: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
function DivergedDialog({ remote, branch, onResult, onHide }: DivergedDialogProps) {
|
||||||
|
const [selected, setSelected] = useState<Resolution | null>(null);
|
||||||
|
|
||||||
|
const handleSubmit = () => {
|
||||||
|
if (selected == null) return;
|
||||||
|
onResult(selected);
|
||||||
|
onHide();
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleCancel = () => {
|
||||||
|
onResult('cancel');
|
||||||
|
onHide();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col gap-4 mb-4">
|
||||||
|
<p className="text-text-subtle">
|
||||||
|
Your local branch has diverged from{' '}
|
||||||
|
<InlineCode>
|
||||||
|
{remote}/{branch}
|
||||||
|
</InlineCode>. How would you like to resolve this?
|
||||||
|
</p>
|
||||||
|
<RadioCards
|
||||||
|
name="diverged-strategy"
|
||||||
|
value={selected}
|
||||||
|
onChange={setSelected}
|
||||||
|
options={[
|
||||||
|
{
|
||||||
|
value: 'merge',
|
||||||
|
label: 'Merge Commit',
|
||||||
|
description: 'Combining local and remote changes into a single merge commit',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 'force_reset',
|
||||||
|
label: 'Force Pull',
|
||||||
|
description: 'Discard local commits and reset to match the remote branch',
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<HStack space={2} justifyContent="start" className="flex-row-reverse">
|
||||||
|
<Button
|
||||||
|
color={selected === 'force_reset' ? 'danger' : 'primary'}
|
||||||
|
disabled={selected == null}
|
||||||
|
onClick={handleSubmit}
|
||||||
|
>
|
||||||
|
{selected != null ? resolutionLabel[selected] : 'Select an option'}
|
||||||
|
</Button>
|
||||||
|
<Button variant="border" onClick={handleCancel}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function promptDivergedStrategy({
|
||||||
|
remote,
|
||||||
|
branch,
|
||||||
|
}: {
|
||||||
|
remote: string;
|
||||||
|
branch: string;
|
||||||
|
}): Promise<DivergedStrategy> {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
showDialog({
|
||||||
|
id: 'git-diverged',
|
||||||
|
title: 'Branches Diverged',
|
||||||
|
hideX: true,
|
||||||
|
size: 'sm',
|
||||||
|
disableBackdropClose: true,
|
||||||
|
onClose: () => resolve('cancel'),
|
||||||
|
render: ({ hide }) =>
|
||||||
|
DivergedDialog({
|
||||||
|
remote,
|
||||||
|
branch,
|
||||||
|
onHide: hide,
|
||||||
|
onResult: resolve,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -26,5 +26,11 @@ export function handlePullResult(r: PullResult) {
|
|||||||
case 'up_to_date':
|
case 'up_to_date':
|
||||||
showToast({ id: 'pull-nothing', message: 'Already up-to-date', color: 'info' });
|
showToast({ id: 'pull-nothing', message: 'Already up-to-date', color: 'info' });
|
||||||
break;
|
break;
|
||||||
|
case 'diverged':
|
||||||
|
// Handled by mutation callback before reaching here
|
||||||
|
break;
|
||||||
|
case 'uncommitted_changes':
|
||||||
|
// Handled by mutation callback before reaching here
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
13
src-web/components/git/uncommitted.tsx
Normal file
13
src-web/components/git/uncommitted.tsx
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
import type { UncommittedChangesStrategy } from '@yaakapp-internal/git';
|
||||||
|
import { showConfirm } from '../../lib/confirm';
|
||||||
|
|
||||||
|
export async function promptUncommittedChangesStrategy(): Promise<UncommittedChangesStrategy> {
|
||||||
|
const confirmed = await showConfirm({
|
||||||
|
id: 'git-uncommitted-changes',
|
||||||
|
title: 'Uncommitted Changes',
|
||||||
|
description: 'You have uncommitted changes. Commit or reset your changes before pulling.',
|
||||||
|
confirmText: 'Reset and Pull',
|
||||||
|
color: 'danger',
|
||||||
|
});
|
||||||
|
return confirmed ? 'reset' : 'cancel';
|
||||||
|
}
|
||||||
@@ -51,10 +51,9 @@ function ActualEventStreamViewer({ response }: Props) {
|
|||||||
<span className="truncate text-xs">{event.data.slice(0, 1000)}</span>
|
<span className="truncate text-xs">{event.data.slice(0, 1000)}</span>
|
||||||
</HStack>
|
</HStack>
|
||||||
}
|
}
|
||||||
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
renderDetail={({ event, index }) => (
|
renderDetail={({ event, index, onClose }) => (
|
||||||
<EventDetail
|
<EventDetail
|
||||||
event={event}
|
event={event}
|
||||||
index={index}
|
index={index}
|
||||||
@@ -62,6 +61,7 @@ function ActualEventStreamViewer({ response }: Props) {
|
|||||||
showingLarge={showingLarge}
|
showingLarge={showingLarge}
|
||||||
setShowLarge={setShowLarge}
|
setShowLarge={setShowLarge}
|
||||||
setShowingLarge={setShowingLarge}
|
setShowingLarge={setShowingLarge}
|
||||||
|
onClose={onClose}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
@@ -75,6 +75,7 @@ function EventDetail({
|
|||||||
showingLarge,
|
showingLarge,
|
||||||
setShowLarge,
|
setShowLarge,
|
||||||
setShowingLarge,
|
setShowingLarge,
|
||||||
|
onClose,
|
||||||
}: {
|
}: {
|
||||||
event: ServerSentEvent;
|
event: ServerSentEvent;
|
||||||
index: number;
|
index: number;
|
||||||
@@ -82,6 +83,7 @@ function EventDetail({
|
|||||||
showingLarge: boolean;
|
showingLarge: boolean;
|
||||||
setShowLarge: (v: boolean) => void;
|
setShowLarge: (v: boolean) => void;
|
||||||
setShowingLarge: (v: boolean) => void;
|
setShowingLarge: (v: boolean) => void;
|
||||||
|
onClose: () => void;
|
||||||
}) {
|
}) {
|
||||||
const language = useMemo<'text' | 'json'>(() => {
|
const language = useMemo<'text' | 'json'>(() => {
|
||||||
if (!event?.data) return 'text';
|
if (!event?.data) return 'text';
|
||||||
@@ -90,7 +92,11 @@ function EventDetail({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col h-full">
|
<div className="flex flex-col h-full">
|
||||||
<EventDetailHeader title="Message Received" prefix={<EventLabels event={event} index={index} />} />
|
<EventDetailHeader
|
||||||
|
title="Message Received"
|
||||||
|
prefix={<EventLabels event={event} index={index} />}
|
||||||
|
onClose={onClose}
|
||||||
|
/>
|
||||||
{!showLarge && event.data.length > 1000 * 1000 ? (
|
{!showLarge && event.data.length > 1000 * 1000 ? (
|
||||||
<VStack space={2} className="italic text-text-subtlest">
|
<VStack space={2} className="italic text-text-subtlest">
|
||||||
Message previews larger than 1MB are hidden
|
Message previews larger than 1MB are hidden
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
import { emit } from '@tauri-apps/api/event';
|
import { emit } from '@tauri-apps/api/event';
|
||||||
import { openUrl } from '@tauri-apps/plugin-opener';
|
import { openUrl } from '@tauri-apps/plugin-opener';
|
||||||
import type { InternalEvent, ShowToastRequest } from '@yaakapp-internal/plugins';
|
import { debounce } from '@yaakapp-internal/lib';
|
||||||
|
import type {
|
||||||
|
FormInput,
|
||||||
|
InternalEvent,
|
||||||
|
JsonPrimitive,
|
||||||
|
ShowToastRequest,
|
||||||
|
} from '@yaakapp-internal/plugins';
|
||||||
import { updateAllPlugins } from '@yaakapp-internal/plugins';
|
import { updateAllPlugins } from '@yaakapp-internal/plugins';
|
||||||
import type {
|
import type {
|
||||||
PluginUpdateNotification,
|
PluginUpdateNotification,
|
||||||
@@ -32,6 +38,9 @@ export function initGlobalListeners() {
|
|||||||
|
|
||||||
listenToTauriEvent('settings', () => openSettings.mutate(null));
|
listenToTauriEvent('settings', () => openSettings.mutate(null));
|
||||||
|
|
||||||
|
// Track active dynamic form dialogs so follow-up input updates can reach them
|
||||||
|
const activeForms = new Map<string, (inputs: FormInput[]) => void>();
|
||||||
|
|
||||||
// Listen for plugin events
|
// Listen for plugin events
|
||||||
listenToTauriEvent<InternalEvent>('plugin_event', async ({ payload: event }) => {
|
listenToTauriEvent<InternalEvent>('plugin_event', async ({ payload: event }) => {
|
||||||
if (event.payload.type === 'prompt_text_request') {
|
if (event.payload.type === 'prompt_text_request') {
|
||||||
@@ -49,26 +58,47 @@ export function initGlobalListeners() {
|
|||||||
};
|
};
|
||||||
await emit(event.id, result);
|
await emit(event.id, result);
|
||||||
} else if (event.payload.type === 'prompt_form_request') {
|
} else if (event.payload.type === 'prompt_form_request') {
|
||||||
|
if (event.replyId != null) {
|
||||||
|
// Follow-up update from plugin runtime — update the active dialog's inputs
|
||||||
|
const updateInputs = activeForms.get(event.replyId);
|
||||||
|
if (updateInputs) {
|
||||||
|
updateInputs(event.payload.inputs);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initial request — show the dialog with bidirectional support
|
||||||
|
const emitFormResponse = (values: Record<string, JsonPrimitive> | null, done: boolean) => {
|
||||||
|
const result: InternalEvent = {
|
||||||
|
id: generateId(),
|
||||||
|
replyId: event.id,
|
||||||
|
pluginName: event.pluginName,
|
||||||
|
pluginRefId: event.pluginRefId,
|
||||||
|
context: event.context,
|
||||||
|
payload: {
|
||||||
|
type: 'prompt_form_response',
|
||||||
|
values,
|
||||||
|
done,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
emit(event.id, result);
|
||||||
|
};
|
||||||
|
|
||||||
const values = await showPromptForm({
|
const values = await showPromptForm({
|
||||||
id: event.payload.id,
|
id: event.payload.id,
|
||||||
title: event.payload.title,
|
title: event.payload.title,
|
||||||
description: event.payload.description,
|
description: event.payload.description,
|
||||||
|
size: event.payload.size,
|
||||||
inputs: event.payload.inputs,
|
inputs: event.payload.inputs,
|
||||||
confirmText: event.payload.confirmText,
|
confirmText: event.payload.confirmText,
|
||||||
cancelText: event.payload.cancelText,
|
cancelText: event.payload.cancelText,
|
||||||
|
onValuesChange: debounce((values) => emitFormResponse(values, false), 150),
|
||||||
|
onInputsUpdated: (cb) => activeForms.set(event.id, cb),
|
||||||
});
|
});
|
||||||
const result: InternalEvent = {
|
|
||||||
id: generateId(),
|
// Clean up and send final response
|
||||||
replyId: event.id,
|
activeForms.delete(event.id);
|
||||||
pluginName: event.pluginName,
|
emitFormResponse(values, true);
|
||||||
pluginRefId: event.pluginRefId,
|
|
||||||
context: event.context,
|
|
||||||
payload: {
|
|
||||||
type: 'prompt_form_response',
|
|
||||||
values,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
await emit(event.id, result);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,21 +1,32 @@
|
|||||||
|
import type { FormInput, JsonPrimitive } from '@yaakapp-internal/plugins';
|
||||||
import type { DialogProps } from '../components/core/Dialog';
|
import type { DialogProps } from '../components/core/Dialog';
|
||||||
import type { PromptProps } from '../components/core/Prompt';
|
import type { PromptProps } from '../components/core/Prompt';
|
||||||
import { Prompt } from '../components/core/Prompt';
|
import { Prompt } from '../components/core/Prompt';
|
||||||
import { showDialog } from './dialog';
|
import { showDialog } from './dialog';
|
||||||
|
|
||||||
type FormArgs = Pick<DialogProps, 'title' | 'description'> &
|
type FormArgs = Pick<DialogProps, 'title' | 'description' | 'size'> &
|
||||||
Omit<PromptProps, 'onClose' | 'onCancel' | 'onResult'> & {
|
Omit<PromptProps, 'onClose' | 'onCancel' | 'onResult'> & {
|
||||||
id: string;
|
id: string;
|
||||||
|
onValuesChange?: (values: Record<string, JsonPrimitive>) => void;
|
||||||
|
onInputsUpdated?: (cb: (inputs: FormInput[]) => void) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
export async function showPromptForm({ id, title, description, ...props }: FormArgs) {
|
export async function showPromptForm({
|
||||||
|
id,
|
||||||
|
title,
|
||||||
|
description,
|
||||||
|
size,
|
||||||
|
onValuesChange,
|
||||||
|
onInputsUpdated,
|
||||||
|
...props
|
||||||
|
}: FormArgs) {
|
||||||
return new Promise((resolve: PromptProps['onResult']) => {
|
return new Promise((resolve: PromptProps['onResult']) => {
|
||||||
showDialog({
|
showDialog({
|
||||||
id,
|
id,
|
||||||
title,
|
title,
|
||||||
description,
|
description,
|
||||||
hideX: true,
|
hideX: true,
|
||||||
size: 'sm',
|
size: size ?? 'sm',
|
||||||
disableBackdropClose: true, // Prevent accidental dismisses
|
disableBackdropClose: true, // Prevent accidental dismisses
|
||||||
onClose: () => {
|
onClose: () => {
|
||||||
// Click backdrop, close, or escape
|
// Click backdrop, close, or escape
|
||||||
@@ -32,6 +43,8 @@ export async function showPromptForm({ id, title, description, ...props }: FormA
|
|||||||
resolve(v);
|
resolve(v);
|
||||||
hide();
|
hide();
|
||||||
},
|
},
|
||||||
|
onValuesChange,
|
||||||
|
onInputsUpdated,
|
||||||
...props,
|
...props,
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ type TauriCmd =
|
|||||||
| 'cmd_call_workspace_action'
|
| 'cmd_call_workspace_action'
|
||||||
| 'cmd_call_folder_action'
|
| 'cmd_call_folder_action'
|
||||||
| 'cmd_check_for_updates'
|
| 'cmd_check_for_updates'
|
||||||
| 'cmd_create_grpc_request'
|
|
||||||
| 'cmd_curl_to_request'
|
| 'cmd_curl_to_request'
|
||||||
| 'cmd_decrypt_template'
|
| 'cmd_decrypt_template'
|
||||||
| 'cmd_default_headers'
|
| 'cmd_default_headers'
|
||||||
@@ -48,9 +47,7 @@ type TauriCmd =
|
|||||||
| 'cmd_save_response'
|
| 'cmd_save_response'
|
||||||
| 'cmd_secure_template'
|
| 'cmd_secure_template'
|
||||||
| 'cmd_send_ephemeral_request'
|
| 'cmd_send_ephemeral_request'
|
||||||
| 'cmd_send_folder'
|
|
||||||
| 'cmd_send_http_request'
|
| 'cmd_send_http_request'
|
||||||
| 'cmd_show_workspace_key'
|
|
||||||
| 'cmd_template_function_summaries'
|
| 'cmd_template_function_summaries'
|
||||||
| 'cmd_template_function_config'
|
| 'cmd_template_function_config'
|
||||||
| 'cmd_template_tokens_to_string';
|
| 'cmd_template_tokens_to_string';
|
||||||
|
|||||||
Reference in New Issue
Block a user