mirror of
https://github.com/mountain-loop/yaak.git
synced 2026-02-27 12:20:24 -05:00
Compare commits
1 Commits
codex/cli-
...
v2025.2.3
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
64f5d973eb |
@@ -1,72 +0,0 @@
|
|||||||
# Claude Context: Detaching Tauri from Yaak
|
|
||||||
|
|
||||||
## Goal
|
|
||||||
Make Yaak runnable as a standalone CLI without Tauri as a dependency. The core Rust crates in `crates/` should be usable independently, while Tauri-specific code lives in `crates-tauri/`.
|
|
||||||
|
|
||||||
## Project Structure
|
|
||||||
```
|
|
||||||
crates/ # Core crates - should NOT depend on Tauri
|
|
||||||
crates-tauri/ # Tauri-specific crates (yaak-app, yaak-tauri-utils, etc.)
|
|
||||||
crates-cli/ # CLI crate (yaak-cli)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Completed Work
|
|
||||||
|
|
||||||
### 1. Folder Restructure
|
|
||||||
- Moved Tauri-dependent app code to `crates-tauri/yaak-app/`
|
|
||||||
- Created `crates-tauri/yaak-tauri-utils/` for shared Tauri utilities (window traits, api_client, error handling)
|
|
||||||
- Created `crates-cli/yaak-cli/` for the standalone CLI
|
|
||||||
|
|
||||||
### 2. Decoupled Crates (no longer depend on Tauri)
|
|
||||||
- **yaak-models**: Uses `init_standalone()` pattern for CLI database access
|
|
||||||
- **yaak-http**: Removed Tauri plugin, HttpConnectionManager initialized in yaak-app setup
|
|
||||||
- **yaak-common**: Only contains Tauri-free utilities (serde, platform)
|
|
||||||
- **yaak-crypto**: Removed Tauri plugin, EncryptionManager initialized in yaak-app setup, commands moved to yaak-app
|
|
||||||
- **yaak-grpc**: Replaced AppHandle with GrpcConfig struct, uses tokio::process::Command instead of Tauri sidecar
|
|
||||||
|
|
||||||
### 3. CLI Implementation
|
|
||||||
- Basic CLI at `crates-cli/yaak-cli/src/main.rs`
|
|
||||||
- Commands: workspaces, requests, send (by ID), get (ad-hoc URL), create
|
|
||||||
- Uses same database as Tauri app via `yaak_models::init_standalone()`
|
|
||||||
|
|
||||||
## Remaining Work
|
|
||||||
|
|
||||||
### Crates Still Depending on Tauri (in `crates/`)
|
|
||||||
1. **yaak-git** (3 files) - Moderate complexity
|
|
||||||
2. **yaak-plugins** (13 files) - **Hardest** - deeply integrated with Tauri for plugin-to-window communication
|
|
||||||
3. **yaak-sync** (4 files) - Moderate complexity
|
|
||||||
4. **yaak-ws** (5 files) - Moderate complexity
|
|
||||||
|
|
||||||
### Pattern for Decoupling
|
|
||||||
1. Remove Tauri plugin `init()` function from the crate
|
|
||||||
2. Move commands to `yaak-app/src/commands.rs` or keep inline in `lib.rs`
|
|
||||||
3. Move extension traits (e.g., `SomethingManagerExt`) to yaak-app or yaak-tauri-utils
|
|
||||||
4. Initialize managers in yaak-app's `.setup()` block
|
|
||||||
5. Remove `tauri` from Cargo.toml dependencies
|
|
||||||
6. Update `crates-tauri/yaak-app/capabilities/default.json` to remove the plugin permission
|
|
||||||
7. Replace `tauri::async_runtime::block_on` with `tokio::runtime::Handle::current().block_on()`
|
|
||||||
|
|
||||||
## Key Files
|
|
||||||
- `crates-tauri/yaak-app/src/lib.rs` - Main Tauri app, setup block initializes managers
|
|
||||||
- `crates-tauri/yaak-app/src/commands.rs` - Migrated Tauri commands
|
|
||||||
- `crates-tauri/yaak-app/src/models_ext.rs` - Database plugin and extension traits
|
|
||||||
- `crates-tauri/yaak-tauri-utils/src/window.rs` - WorkspaceWindowTrait for window state
|
|
||||||
- `crates/yaak-models/src/lib.rs` - Contains `init_standalone()` for CLI usage
|
|
||||||
|
|
||||||
## Git Branch
|
|
||||||
Working on `detach-tauri` branch.
|
|
||||||
|
|
||||||
## Recent Commits
|
|
||||||
```
|
|
||||||
c40cff40 Remove Tauri dependencies from yaak-crypto and yaak-grpc
|
|
||||||
df495f1d Move Tauri utilities from yaak-common to yaak-tauri-utils
|
|
||||||
481e0273 Remove Tauri dependencies from yaak-http and yaak-common
|
|
||||||
10568ac3 Add HTTP request sending to yaak-cli
|
|
||||||
bcb7d600 Add yaak-cli stub with basic database access
|
|
||||||
e718a5f1 Refactor models_ext to use init_standalone from yaak-models
|
|
||||||
```
|
|
||||||
|
|
||||||
## Testing
|
|
||||||
- Run `cargo check -p <crate>` to verify a crate builds without Tauri
|
|
||||||
- Run `npm run app-dev` to test the Tauri app still works
|
|
||||||
- Run `cargo run -p yaak-cli -- --help` to test the CLI
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
---
|
|
||||||
description: Review a PR in a new worktree
|
|
||||||
allowed-tools: Bash(git worktree:*), Bash(gh pr:*), Bash(git branch:*)
|
|
||||||
---
|
|
||||||
|
|
||||||
Check out a GitHub pull request for review.
|
|
||||||
|
|
||||||
## Usage
|
|
||||||
|
|
||||||
```
|
|
||||||
/check-out-pr <PR_NUMBER>
|
|
||||||
```
|
|
||||||
|
|
||||||
## What to do
|
|
||||||
|
|
||||||
1. If no PR number is provided, list all open pull requests and ask the user to select one
|
|
||||||
2. Get PR information using `gh pr view <PR_NUMBER> --json number,headRefName`
|
|
||||||
3. **Ask the user** whether they want to:
|
|
||||||
- **A) Check out in current directory** — simple `gh pr checkout <PR_NUMBER>`
|
|
||||||
- **B) Create a new worktree** — isolated copy at `../yaak-worktrees/pr-<PR_NUMBER>`
|
|
||||||
4. Follow the appropriate path below
|
|
||||||
|
|
||||||
## Option A: Check out in current directory
|
|
||||||
|
|
||||||
1. Run `gh pr checkout <PR_NUMBER>`
|
|
||||||
2. Inform the user which branch they're now on
|
|
||||||
|
|
||||||
## Option B: Create a new worktree
|
|
||||||
|
|
||||||
1. Create a new worktree at `../yaak-worktrees/pr-<PR_NUMBER>` using `git worktree add` with a timeout of at least 300000ms (5 minutes) since the post-checkout hook runs a bootstrap script
|
|
||||||
2. Checkout the PR branch in the new worktree using `gh pr checkout <PR_NUMBER>`
|
|
||||||
3. The post-checkout hook will automatically:
|
|
||||||
- Create `.env.local` with unique ports
|
|
||||||
- Copy editor config folders
|
|
||||||
- Run `npm install && npm run bootstrap`
|
|
||||||
4. Inform the user:
|
|
||||||
- Where the worktree was created
|
|
||||||
- What ports were assigned
|
|
||||||
- How to access it (cd command)
|
|
||||||
- How to run the dev server
|
|
||||||
- How to remove the worktree when done
|
|
||||||
|
|
||||||
### Example worktree output
|
|
||||||
|
|
||||||
```
|
|
||||||
Created worktree for PR #123 at ../yaak-worktrees/pr-123
|
|
||||||
Branch: feature-auth
|
|
||||||
Ports: Vite (1421), MCP (64344)
|
|
||||||
|
|
||||||
To start working:
|
|
||||||
cd ../yaak-worktrees/pr-123
|
|
||||||
npm run app-dev
|
|
||||||
|
|
||||||
To remove when done:
|
|
||||||
git worktree remove ../yaak-worktrees/pr-123
|
|
||||||
```
|
|
||||||
|
|
||||||
## Error Handling
|
|
||||||
|
|
||||||
- If the PR doesn't exist, show a helpful error
|
|
||||||
- If the worktree already exists, inform the user and ask if they want to remove and recreate it
|
|
||||||
- If `gh` CLI is not available, inform the user to install it
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
---
|
|
||||||
description: Generate formatted release notes for Yaak releases
|
|
||||||
allowed-tools: Bash(git tag:*)
|
|
||||||
---
|
|
||||||
|
|
||||||
Generate formatted release notes for Yaak releases by analyzing git history and pull request descriptions.
|
|
||||||
|
|
||||||
## What to do
|
|
||||||
|
|
||||||
1. Identifies the version tag and previous version
|
|
||||||
2. Retrieves all commits between versions
|
|
||||||
- If the version is a beta version, it retrieves commits between the beta version and previous beta version
|
|
||||||
- If the version is a stable version, it retrieves commits between the stable version and the previous stable version
|
|
||||||
3. Fetches PR descriptions for linked issues to find:
|
|
||||||
- Feedback URLs (feedback.yaak.app)
|
|
||||||
- Additional context and descriptions
|
|
||||||
- Installation links for plugins
|
|
||||||
4. Formats the release notes using the standard Yaak format:
|
|
||||||
- Changelog badge at the top
|
|
||||||
- Bulleted list of changes with PR links
|
|
||||||
- Feedback links where available
|
|
||||||
- Full changelog comparison link at the bottom
|
|
||||||
|
|
||||||
## Output Format
|
|
||||||
|
|
||||||
The skill generates markdown-formatted release notes following this structure:
|
|
||||||
|
|
||||||
```markdown
|
|
||||||
[](https://yaak.app/changelog/VERSION)
|
|
||||||
|
|
||||||
- Feature/fix description in by @username [#123](https://github.com/mountain-loop/yaak/pull/123)
|
|
||||||
- [Linked feedback item](https://feedback.yaak.app/p/item) by @username in [#456](https://github.com/mountain-loop/yaak/pull/456)
|
|
||||||
- A simple item that doesn't have a feedback or PR link
|
|
||||||
|
|
||||||
**Full Changelog**: https://github.com/mountain-loop/yaak/compare/vPREV...vCURRENT
|
|
||||||
```
|
|
||||||
|
|
||||||
**IMPORTANT**: Always add a blank lines around the markdown code fence and output the markdown code block last
|
|
||||||
**IMPORTANT**: PRs by `@gschier` should not mention the @username
|
|
||||||
|
|
||||||
## After Generating Release Notes
|
|
||||||
|
|
||||||
After outputting the release notes, ask the user if they would like to create a draft GitHub release with these notes. If they confirm, create the release using:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
gh release create <tag> --draft --prerelease --title "Release <version>" --notes '<release notes>'
|
|
||||||
```
|
|
||||||
|
|
||||||
**IMPORTANT**: The release title format is "Release XXXX" where XXXX is the version WITHOUT the `v` prefix. For example, tag `v2026.2.1-beta.1` gets title "Release 2026.2.1-beta.1".
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
# Project Rules
|
|
||||||
|
|
||||||
## General Development
|
|
||||||
|
|
||||||
- **NEVER** commit or push without explicit confirmation
|
|
||||||
|
|
||||||
## Build and Lint
|
|
||||||
|
|
||||||
- **ALWAYS** run `npm run lint` after modifying TypeScript or JavaScript files
|
|
||||||
- Run `npm run bootstrap` after changing plugin runtime or MCP server code
|
|
||||||
|
|
||||||
## Plugin System
|
|
||||||
|
|
||||||
### Backend Constraints
|
|
||||||
|
|
||||||
- Always use `UpdateSource::Plugin` when calling database methods from plugin events
|
|
||||||
- Never send timestamps (`createdAt`, `updatedAt`) from TypeScript - Rust backend controls these
|
|
||||||
- Backend uses `NaiveDateTime` (no timezone) so avoid sending ISO timestamp strings
|
|
||||||
|
|
||||||
### MCP Server
|
|
||||||
|
|
||||||
- MCP server has **no active window context** - cannot call `window.workspaceId()`
|
|
||||||
- Get workspace ID from `workspaceCtx.yaak.workspace.list()` instead
|
|
||||||
|
|
||||||
## Rust Type Generation
|
|
||||||
|
|
||||||
- Run `cargo test --package yaak-plugins` (and for other crates) to regenerate TypeScript bindings after modifying Rust event types
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
# Worktree Management Skill
|
|
||||||
|
|
||||||
## Creating Worktrees
|
|
||||||
|
|
||||||
When creating git worktrees for this project, ALWAYS use the path format:
|
|
||||||
```
|
|
||||||
../yaak-worktrees/<NAME>
|
|
||||||
```
|
|
||||||
|
|
||||||
For example:
|
|
||||||
- `git worktree add ../yaak-worktrees/feature-auth`
|
|
||||||
- `git worktree add ../yaak-worktrees/bugfix-login`
|
|
||||||
- `git worktree add ../yaak-worktrees/refactor-api`
|
|
||||||
|
|
||||||
## What Happens Automatically
|
|
||||||
|
|
||||||
The post-checkout hook will automatically:
|
|
||||||
1. Create `.env.local` with unique ports (YAAK_DEV_PORT and YAAK_PLUGIN_MCP_SERVER_PORT)
|
|
||||||
2. Copy gitignored editor config folders (.zed, .idea, etc.)
|
|
||||||
3. Run `npm install && npm run bootstrap`
|
|
||||||
|
|
||||||
## Deleting Worktrees
|
|
||||||
|
|
||||||
```bash
|
|
||||||
git worktree remove ../yaak-worktrees/<NAME>
|
|
||||||
```
|
|
||||||
|
|
||||||
## Port Assignments
|
|
||||||
|
|
||||||
- Main worktree: 1420 (Vite), 64343 (MCP)
|
|
||||||
- First worktree: 1421, 64344
|
|
||||||
- Second worktree: 1422, 64345
|
|
||||||
- etc.
|
|
||||||
|
|
||||||
Each worktree can run `npm run app-dev` simultaneously without conflicts.
|
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
---
|
|
||||||
name: release-check-out-pr
|
|
||||||
description: Check out a GitHub pull request for review in this repo, either in the current directory or in a new isolated worktree at ../yaak-worktrees/pr-<PR_NUMBER>. Use when asked to run or replace the old Claude check-out-pr command.
|
|
||||||
---
|
|
||||||
|
|
||||||
# Check Out PR
|
|
||||||
|
|
||||||
Check out a PR by number and let the user choose between current-directory checkout and isolated worktree checkout.
|
|
||||||
|
|
||||||
## Workflow
|
|
||||||
|
|
||||||
1. Confirm `gh` CLI is available.
|
|
||||||
2. If no PR number is provided, list open PRs (`gh pr list`) and ask the user to choose one.
|
|
||||||
3. Read PR metadata:
|
|
||||||
- `gh pr view <PR_NUMBER> --json number,headRefName`
|
|
||||||
4. Ask the user to choose:
|
|
||||||
- Option A: check out in the current directory
|
|
||||||
- Option B: create a new worktree at `../yaak-worktrees/pr-<PR_NUMBER>`
|
|
||||||
|
|
||||||
## Option A: Current Directory
|
|
||||||
|
|
||||||
1. Run:
|
|
||||||
- `gh pr checkout <PR_NUMBER>`
|
|
||||||
2. Report the checked-out branch.
|
|
||||||
|
|
||||||
## Option B: New Worktree
|
|
||||||
|
|
||||||
1. Use path:
|
|
||||||
- `../yaak-worktrees/pr-<PR_NUMBER>`
|
|
||||||
2. Create the worktree with a timeout of at least 5 minutes because checkout hooks run bootstrap.
|
|
||||||
3. In the new worktree, run:
|
|
||||||
- `gh pr checkout <PR_NUMBER>`
|
|
||||||
4. Report:
|
|
||||||
- Worktree path
|
|
||||||
- Assigned ports from `.env.local` if present
|
|
||||||
- How to start work:
|
|
||||||
- `cd ../yaak-worktrees/pr-<PR_NUMBER>`
|
|
||||||
- `npm run app-dev`
|
|
||||||
- How to remove when done:
|
|
||||||
- `git worktree remove ../yaak-worktrees/pr-<PR_NUMBER>`
|
|
||||||
|
|
||||||
## Error Handling
|
|
||||||
|
|
||||||
- If PR does not exist, show a clear error.
|
|
||||||
- If worktree already exists, ask whether to reuse it or remove/recreate it.
|
|
||||||
- If `gh` is missing, instruct the user to install/authenticate it.
|
|
||||||
@@ -1,48 +0,0 @@
|
|||||||
---
|
|
||||||
name: release-generate-release-notes
|
|
||||||
description: Generate Yaak release notes from git history and PR metadata, including feedback links and full changelog compare links. Use when asked to run or replace the old Claude generate-release-notes command.
|
|
||||||
---
|
|
||||||
|
|
||||||
# Generate Release Notes
|
|
||||||
|
|
||||||
Generate formatted markdown release notes for a Yaak tag.
|
|
||||||
|
|
||||||
## Workflow
|
|
||||||
|
|
||||||
1. Determine target tag.
|
|
||||||
2. Determine previous comparable tag:
|
|
||||||
- Beta tag: compare against previous beta (if the root version is the same) or stable tag.
|
|
||||||
- Stable tag: compare against previous stable tag.
|
|
||||||
3. Collect commits in range:
|
|
||||||
- `git log --oneline <prev_tag>..<target_tag>`
|
|
||||||
4. For linked PRs, fetch metadata:
|
|
||||||
- `gh pr view <PR_NUMBER> --json number,title,body,author,url`
|
|
||||||
5. Extract useful details:
|
|
||||||
- Feedback URLs (`feedback.yaak.app`)
|
|
||||||
- Plugin install links or other notable context
|
|
||||||
6. Format notes using Yaak style:
|
|
||||||
- Changelog badge at top
|
|
||||||
- Bulleted items with PR links where available
|
|
||||||
- Feedback links where available
|
|
||||||
- Full changelog compare link at bottom
|
|
||||||
|
|
||||||
## Formatting Rules
|
|
||||||
|
|
||||||
- Wrap final notes in a markdown code fence.
|
|
||||||
- Keep a blank line before and after the code fence.
|
|
||||||
- Output the markdown code block last.
|
|
||||||
- Do not append `by @gschier` for PRs authored by `@gschier`.
|
|
||||||
|
|
||||||
## Release Creation Prompt
|
|
||||||
|
|
||||||
After producing notes, ask whether to create a draft GitHub release.
|
|
||||||
|
|
||||||
If confirmed and release does not yet exist, run:
|
|
||||||
|
|
||||||
`gh release create <tag> --draft --prerelease --title "Release <version_without_v>" --notes '<release notes>'`
|
|
||||||
|
|
||||||
If a draft release for the tag already exists, update it instead:
|
|
||||||
|
|
||||||
`gh release edit <tag> --title "Release <version_without_v>" --notes-file <path_to_notes>`
|
|
||||||
|
|
||||||
Use title format `Release <version_without_v>`, e.g. `v2026.2.1-beta.1` -> `Release 2026.2.1-beta.1`.
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
---
|
|
||||||
name: worktree-management
|
|
||||||
description: Manage Yaak git worktrees using the standard ../yaak-worktrees/<NAME> layout, including creation, removal, and expected automatic setup behavior and port assignments.
|
|
||||||
---
|
|
||||||
|
|
||||||
# Worktree Management
|
|
||||||
|
|
||||||
Use the Yaak-standard worktree path layout and lifecycle commands.
|
|
||||||
|
|
||||||
## Path Convention
|
|
||||||
|
|
||||||
Always create worktrees under:
|
|
||||||
|
|
||||||
`../yaak-worktrees/<NAME>`
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
- `git worktree add ../yaak-worktrees/feature-auth`
|
|
||||||
- `git worktree add ../yaak-worktrees/bugfix-login`
|
|
||||||
- `git worktree add ../yaak-worktrees/refactor-api`
|
|
||||||
|
|
||||||
## Automatic Setup After Checkout
|
|
||||||
|
|
||||||
Project git hooks automatically:
|
|
||||||
1. Create `.env.local` with unique `YAAK_DEV_PORT` and `YAAK_PLUGIN_MCP_SERVER_PORT`
|
|
||||||
2. Copy gitignored editor config folders
|
|
||||||
3. Run `npm install && npm run bootstrap`
|
|
||||||
|
|
||||||
## Remove Worktree
|
|
||||||
|
|
||||||
`git worktree remove ../yaak-worktrees/<NAME>`
|
|
||||||
|
|
||||||
## Port Pattern
|
|
||||||
|
|
||||||
- Main worktree: Vite `1420`, MCP `64343`
|
|
||||||
- First extra worktree: `1421`, `64344`
|
|
||||||
- Second extra worktree: `1422`, `64345`
|
|
||||||
- Continue incrementally for additional worktrees
|
|
||||||
6
.eslintignore
Normal file
6
.eslintignore
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
node_modules/
|
||||||
|
dist/
|
||||||
|
.eslintrc.cjs
|
||||||
|
.prettierrc.cjs
|
||||||
|
src-web/postcss.config.cjs
|
||||||
|
src-web/vite.config.ts
|
||||||
49
.eslintrc.cjs
Normal file
49
.eslintrc.cjs
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
module.exports = {
|
||||||
|
extends: [
|
||||||
|
'eslint:recommended',
|
||||||
|
'plugin:react/recommended',
|
||||||
|
'plugin:react-hooks/recommended',
|
||||||
|
'plugin:import/recommended',
|
||||||
|
'plugin:jsx-a11y/recommended',
|
||||||
|
'plugin:@typescript-eslint/recommended',
|
||||||
|
'eslint-config-prettier',
|
||||||
|
],
|
||||||
|
plugins: ['react-refresh'],
|
||||||
|
parser: '@typescript-eslint/parser',
|
||||||
|
parserOptions: {
|
||||||
|
project: ['./tsconfig.json'],
|
||||||
|
},
|
||||||
|
ignorePatterns: [
|
||||||
|
'scripts/**/*',
|
||||||
|
'packages/plugin-runtime/**/*',
|
||||||
|
'packages/plugin-runtime-types/**/*',
|
||||||
|
'src-tauri/**/*',
|
||||||
|
'src-web/tailwind.config.cjs',
|
||||||
|
'src-web/vite.config.ts',
|
||||||
|
],
|
||||||
|
settings: {
|
||||||
|
react: {
|
||||||
|
version: 'detect',
|
||||||
|
},
|
||||||
|
'import/resolver': {
|
||||||
|
node: {
|
||||||
|
paths: ['src-web'],
|
||||||
|
extensions: ['.ts', '.tsx'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
rules: {
|
||||||
|
'react-refresh/only-export-components': 'error',
|
||||||
|
'jsx-a11y/no-autofocus': 'off',
|
||||||
|
'react/react-in-jsx-scope': 'off',
|
||||||
|
'import/no-unresolved': 'off',
|
||||||
|
'@typescript-eslint/consistent-type-imports': [
|
||||||
|
'error',
|
||||||
|
{
|
||||||
|
prefer: 'type-imports',
|
||||||
|
disallowTypeAnnotations: true,
|
||||||
|
fixStyle: 'separate-type-imports',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
};
|
||||||
9
.gitattributes
vendored
9
.gitattributes
vendored
@@ -1,7 +1,2 @@
|
|||||||
crates-tauri/yaak-app/vendored/**/* linguist-generated=true
|
src-tauri/vendored/**/* linguist-generated=true
|
||||||
crates-tauri/yaak-app/gen/schemas/**/* linguist-generated=true
|
src-tauri/gen/schemas/**/* linguist-generated=true
|
||||||
**/bindings/* linguist-generated=true
|
|
||||||
crates/yaak-templates/pkg/* linguist-generated=true
|
|
||||||
|
|
||||||
# Ensure consistent line endings for test files that check exact content
|
|
||||||
crates/yaak-http/tests/test.txt text eol=lf
|
|
||||||
|
|||||||
12
.github/FUNDING.yml
vendored
12
.github/FUNDING.yml
vendored
@@ -1,3 +1,15 @@
|
|||||||
# These are supported funding model platforms
|
# These are supported funding model platforms
|
||||||
|
|
||||||
github: gschier
|
github: gschier
|
||||||
|
patreon: # Replace with a single Patreon username
|
||||||
|
open_collective: # Replace with a single Open Collective username
|
||||||
|
ko_fi: # Replace with a single Ko-fi username
|
||||||
|
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
|
||||||
|
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
|
||||||
|
liberapay: # Replace with a single Liberapay username
|
||||||
|
issuehunt: # Replace with a single IssueHunt username
|
||||||
|
lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry
|
||||||
|
polar: # Replace with a single Polar username
|
||||||
|
buy_me_a_coffee: # Replace with a single Buy Me a Coffee username
|
||||||
|
thanks_dev: # Replace with a single thanks.dev username
|
||||||
|
custom: https://yaak.app/pricing
|
||||||
|
|||||||
18
.github/pull_request_template.md
vendored
18
.github/pull_request_template.md
vendored
@@ -1,18 +0,0 @@
|
|||||||
## Summary
|
|
||||||
|
|
||||||
<!-- Describe the bug and the fix in 1-3 sentences. -->
|
|
||||||
|
|
||||||
## Submission
|
|
||||||
|
|
||||||
- [ ] This PR is a bug fix or small-scope improvement.
|
|
||||||
- [ ] If this PR is not a bug fix or small-scope improvement, I linked an approved feedback item below.
|
|
||||||
- [ ] I have read and followed [`CONTRIBUTING.md`](CONTRIBUTING.md).
|
|
||||||
- [ ] I tested this change locally.
|
|
||||||
- [ ] I added or updated tests when reasonable.
|
|
||||||
|
|
||||||
Approved feedback item (required if not a bug fix or small-scope improvement):
|
|
||||||
<!-- https://yaak.app/feedback/... -->
|
|
||||||
|
|
||||||
## Related
|
|
||||||
|
|
||||||
<!-- Link related issues, discussions, or feedback items. -->
|
|
||||||
18
.github/workflows/ci-js.yml
vendored
Normal file
18
.github/workflows/ci-js.yml
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches: [develop]
|
||||||
|
|
||||||
|
name: CI (JS)
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
name: Lint/Test
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 20
|
||||||
|
- run: npm ci
|
||||||
|
- run: npm run lint
|
||||||
|
- run: npm test
|
||||||
36
.github/workflows/ci-rust.yml
vendored
Normal file
36
.github/workflows/ci-rust.yml
vendored
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches: [develop]
|
||||||
|
paths:
|
||||||
|
- src-tauri/**
|
||||||
|
- .github/workflows/**
|
||||||
|
|
||||||
|
name: CI (Rust)
|
||||||
|
|
||||||
|
defaults:
|
||||||
|
run:
|
||||||
|
working-directory: src-tauri
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
name: Check/Test
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- run: |
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install -y libwebkit2gtk-4.1-dev
|
||||||
|
- uses: dtolnay/rust-toolchain@stable
|
||||||
|
- uses: actions/cache@v3
|
||||||
|
continue-on-error: false
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
~/.cargo/bin/
|
||||||
|
~/.cargo/registry/index/
|
||||||
|
~/.cargo/registry/cache/
|
||||||
|
~/.cargo/git/db/
|
||||||
|
target/
|
||||||
|
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
||||||
|
restore-keys: ${{ runner.os }}-cargo-
|
||||||
|
- run: cargo check --all
|
||||||
|
- run: cargo test --all
|
||||||
30
.github/workflows/ci.yml
vendored
30
.github/workflows/ci.yml
vendored
@@ -1,30 +0,0 @@
|
|||||||
on:
|
|
||||||
pull_request:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- main
|
|
||||||
|
|
||||||
name: Lint and Test
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
test:
|
|
||||||
name: Lint/Test
|
|
||||||
runs-on: macos-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- uses: actions/setup-node@v4
|
|
||||||
- uses: dtolnay/rust-toolchain@stable
|
|
||||||
- uses: Swatinem/rust-cache@v2
|
|
||||||
with:
|
|
||||||
shared-key: ci
|
|
||||||
cache-on-failure: true
|
|
||||||
|
|
||||||
- run: npm ci
|
|
||||||
- run: npm run bootstrap
|
|
||||||
- run: npm run lint
|
|
||||||
- name: Run JS Tests
|
|
||||||
run: npm test
|
|
||||||
- name: Run Rust Tests
|
|
||||||
run: cargo test --all
|
|
||||||
50
.github/workflows/claude.yml
vendored
50
.github/workflows/claude.yml
vendored
@@ -1,50 +0,0 @@
|
|||||||
name: Claude Code
|
|
||||||
|
|
||||||
on:
|
|
||||||
issue_comment:
|
|
||||||
types: [created]
|
|
||||||
pull_request_review_comment:
|
|
||||||
types: [created]
|
|
||||||
issues:
|
|
||||||
types: [opened, assigned]
|
|
||||||
pull_request_review:
|
|
||||||
types: [submitted]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
claude:
|
|
||||||
if: |
|
|
||||||
(github.event_name == 'issue_comment' && contains(github.event.comment.body, '@claude')) ||
|
|
||||||
(github.event_name == 'pull_request_review_comment' && contains(github.event.comment.body, '@claude')) ||
|
|
||||||
(github.event_name == 'pull_request_review' && contains(github.event.review.body, '@claude')) ||
|
|
||||||
(github.event_name == 'issues' && (contains(github.event.issue.body, '@claude') || contains(github.event.issue.title, '@claude')))
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
pull-requests: read
|
|
||||||
issues: read
|
|
||||||
id-token: write
|
|
||||||
actions: read # Required for Claude to read CI results on PRs
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
fetch-depth: 1
|
|
||||||
|
|
||||||
- name: Run Claude Code
|
|
||||||
id: claude
|
|
||||||
uses: anthropics/claude-code-action@v1
|
|
||||||
with:
|
|
||||||
claude_code_oauth_token: ${{ secrets.CLAUDE_CODE_OAUTH_TOKEN }}
|
|
||||||
|
|
||||||
# This is an optional setting that allows Claude to read CI results on PRs
|
|
||||||
additional_permissions: |
|
|
||||||
actions: read
|
|
||||||
|
|
||||||
# Optional: Give a custom prompt to Claude. If this is not specified, Claude will perform the instructions specified in the comment that tagged it.
|
|
||||||
# prompt: 'Update the pull request description to include a summary of changes.'
|
|
||||||
|
|
||||||
# Optional: Add claude_args to customize behavior and configuration
|
|
||||||
# See https://github.com/anthropics/claude-code-action/blob/main/docs/usage.md
|
|
||||||
# or https://code.claude.com/docs/en/cli-reference for available options
|
|
||||||
# claude_args: '--allowed-tools Bash(gh pr:*)'
|
|
||||||
|
|
||||||
52
.github/workflows/flathub.yml
vendored
52
.github/workflows/flathub.yml
vendored
@@ -1,52 +0,0 @@
|
|||||||
name: Update Flathub
|
|
||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
update-flathub:
|
|
||||||
name: Update Flathub manifest
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
# Only run for stable releases (skip betas/pre-releases)
|
|
||||||
if: ${{ !github.event.release.prerelease }}
|
|
||||||
steps:
|
|
||||||
- name: Checkout app repo
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Checkout Flathub repo
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
repository: flathub/app.yaak.Yaak
|
|
||||||
token: ${{ secrets.FLATHUB_TOKEN }}
|
|
||||||
path: flathub-repo
|
|
||||||
|
|
||||||
- name: Set up Python
|
|
||||||
uses: actions/setup-python@v5
|
|
||||||
with:
|
|
||||||
python-version: "3.12"
|
|
||||||
|
|
||||||
- name: Set up Node.js
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: "22"
|
|
||||||
|
|
||||||
- name: Install source generators
|
|
||||||
run: |
|
|
||||||
pip install flatpak-node-generator tomlkit aiohttp
|
|
||||||
git clone --depth 1 https://github.com/flatpak/flatpak-builder-tools flatpak/flatpak-builder-tools
|
|
||||||
|
|
||||||
- name: Run update-manifest.sh
|
|
||||||
run: bash flatpak/update-manifest.sh "${{ github.event.release.tag_name }}" flathub-repo
|
|
||||||
|
|
||||||
- name: Commit and push to Flathub
|
|
||||||
working-directory: flathub-repo
|
|
||||||
run: |
|
|
||||||
git config user.name "github-actions[bot]"
|
|
||||||
git config user.email "github-actions[bot]@users.noreply.github.com"
|
|
||||||
git add -A
|
|
||||||
git diff --cached --quiet && echo "No changes to commit" && exit 0
|
|
||||||
git commit -m "Update to ${{ github.event.release.tag_name }}"
|
|
||||||
git push
|
|
||||||
59
.github/workflows/release-api-npm.yml
vendored
59
.github/workflows/release-api-npm.yml
vendored
@@ -1,59 +0,0 @@
|
|||||||
name: Release API to NPM
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags: [yaak-api-*]
|
|
||||||
workflow_dispatch:
|
|
||||||
inputs:
|
|
||||||
version:
|
|
||||||
description: API version to publish (for example 0.9.0 or v0.9.0)
|
|
||||||
required: true
|
|
||||||
type: string
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
publish-npm:
|
|
||||||
name: Publish @yaakapp/api
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
id-token: write
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Setup Node
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: lts/*
|
|
||||||
registry-url: https://registry.npmjs.org
|
|
||||||
|
|
||||||
- name: Install dependencies
|
|
||||||
run: npm ci
|
|
||||||
|
|
||||||
- name: Set @yaakapp/api version
|
|
||||||
shell: bash
|
|
||||||
env:
|
|
||||||
WORKFLOW_VERSION: ${{ inputs.version }}
|
|
||||||
run: |
|
|
||||||
set -euo pipefail
|
|
||||||
if [ "${{ github.event_name }}" = "workflow_dispatch" ]; then
|
|
||||||
VERSION="$WORKFLOW_VERSION"
|
|
||||||
else
|
|
||||||
VERSION="${GITHUB_REF_NAME#yaak-api-}"
|
|
||||||
fi
|
|
||||||
VERSION="${VERSION#v}"
|
|
||||||
echo "Preparing @yaakapp/api version: $VERSION"
|
|
||||||
cd packages/plugin-runtime-types
|
|
||||||
npm version "$VERSION" --no-git-tag-version --allow-same-version
|
|
||||||
|
|
||||||
- name: Build @yaakapp/api
|
|
||||||
working-directory: packages/plugin-runtime-types
|
|
||||||
run: npm run build
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/api
|
|
||||||
working-directory: packages/plugin-runtime-types
|
|
||||||
run: npm publish --provenance --access public
|
|
||||||
179
.github/workflows/release-app.yml
vendored
179
.github/workflows/release-app.yml
vendored
@@ -1,179 +0,0 @@
|
|||||||
name: Release App Artifacts
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags: [v*]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build-artifacts:
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
|
|
||||||
name: Build
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
- platform: "macos-latest" # for Arm-based Macs (M1 and above).
|
|
||||||
args: "--target aarch64-apple-darwin"
|
|
||||||
yaak_arch: "arm64"
|
|
||||||
os: "macos"
|
|
||||||
targets: "aarch64-apple-darwin"
|
|
||||||
- platform: "macos-latest" # for Intel-based Macs.
|
|
||||||
args: "--target x86_64-apple-darwin"
|
|
||||||
yaak_arch: "x64"
|
|
||||||
os: "macos"
|
|
||||||
targets: "x86_64-apple-darwin"
|
|
||||||
- platform: "ubuntu-22.04"
|
|
||||||
args: ""
|
|
||||||
yaak_arch: "x64"
|
|
||||||
os: "ubuntu"
|
|
||||||
targets: ""
|
|
||||||
- platform: "ubuntu-22.04-arm"
|
|
||||||
args: ""
|
|
||||||
yaak_arch: "arm64"
|
|
||||||
os: "ubuntu"
|
|
||||||
targets: ""
|
|
||||||
- platform: "windows-latest"
|
|
||||||
args: ""
|
|
||||||
yaak_arch: "x64"
|
|
||||||
os: "windows"
|
|
||||||
targets: ""
|
|
||||||
# Windows ARM64
|
|
||||||
- platform: "windows-latest"
|
|
||||||
args: "--target aarch64-pc-windows-msvc"
|
|
||||||
yaak_arch: "arm64"
|
|
||||||
os: "windows"
|
|
||||||
targets: "aarch64-pc-windows-msvc"
|
|
||||||
runs-on: ${{ matrix.platform }}
|
|
||||||
timeout-minutes: 40
|
|
||||||
steps:
|
|
||||||
- name: Checkout yaakapp/app
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Setup Node
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
|
|
||||||
- name: install Rust stable
|
|
||||||
uses: dtolnay/rust-toolchain@stable
|
|
||||||
with:
|
|
||||||
targets: ${{ matrix.targets }}
|
|
||||||
|
|
||||||
- uses: Swatinem/rust-cache@v2
|
|
||||||
with:
|
|
||||||
shared-key: ci
|
|
||||||
cache-on-failure: true
|
|
||||||
|
|
||||||
- name: install dependencies (Linux only)
|
|
||||||
if: matrix.os == 'ubuntu'
|
|
||||||
run: |
|
|
||||||
sudo apt-get update
|
|
||||||
sudo apt-get install -y libwebkit2gtk-4.1-dev libappindicator3-dev librsvg2-dev patchelf xdg-utils
|
|
||||||
|
|
||||||
- name: Install Protoc for plugin-runtime
|
|
||||||
uses: arduino/setup-protoc@v3
|
|
||||||
with:
|
|
||||||
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
|
|
||||||
- name: Install trusted-signing-cli (Windows only)
|
|
||||||
if: matrix.os == 'windows'
|
|
||||||
shell: pwsh
|
|
||||||
run: |
|
|
||||||
$ErrorActionPreference = 'Stop'
|
|
||||||
$dir = "$env:USERPROFILE\trusted-signing"
|
|
||||||
New-Item -ItemType Directory -Force -Path $dir | Out-Null
|
|
||||||
$url = "https://github.com/Levminer/trusted-signing-cli/releases/download/0.8.0/trusted-signing-cli.exe"
|
|
||||||
$exe = Join-Path $dir "trusted-signing-cli.exe"
|
|
||||||
Invoke-WebRequest -Uri $url -OutFile $exe
|
|
||||||
echo $dir >> $env:GITHUB_PATH
|
|
||||||
& $exe --version
|
|
||||||
|
|
||||||
- run: npm ci
|
|
||||||
- run: npm run bootstrap
|
|
||||||
env:
|
|
||||||
YAAK_TARGET_ARCH: ${{ matrix.yaak_arch }}
|
|
||||||
- run: npm run lint
|
|
||||||
- name: Run JS Tests
|
|
||||||
run: npm test
|
|
||||||
- name: Run Rust Tests
|
|
||||||
run: cargo test --all
|
|
||||||
|
|
||||||
- name: Set version
|
|
||||||
run: npm run replace-version
|
|
||||||
env:
|
|
||||||
YAAK_VERSION: ${{ github.ref_name }}
|
|
||||||
|
|
||||||
- name: Sign vendored binaries (macOS only)
|
|
||||||
if: matrix.os == 'macos'
|
|
||||||
env:
|
|
||||||
APPLE_CERTIFICATE: ${{ secrets.APPLE_CERTIFICATE }}
|
|
||||||
APPLE_CERTIFICATE_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }}
|
|
||||||
APPLE_SIGNING_IDENTITY: ${{ secrets.APPLE_SIGNING_IDENTITY }}
|
|
||||||
KEYCHAIN_PASSWORD: ${{ secrets.KEYCHAIN_PASSWORD }}
|
|
||||||
run: |
|
|
||||||
# Create keychain
|
|
||||||
KEYCHAIN_PATH=$RUNNER_TEMP/app-signing.keychain-db
|
|
||||||
security create-keychain -p "$KEYCHAIN_PASSWORD" $KEYCHAIN_PATH
|
|
||||||
security set-keychain-settings -lut 21600 $KEYCHAIN_PATH
|
|
||||||
security unlock-keychain -p "$KEYCHAIN_PASSWORD" $KEYCHAIN_PATH
|
|
||||||
|
|
||||||
# Import certificate
|
|
||||||
echo "$APPLE_CERTIFICATE" | base64 --decode > certificate.p12
|
|
||||||
security import certificate.p12 -P "$APPLE_CERTIFICATE_PASSWORD" -A -t cert -f pkcs12 -k $KEYCHAIN_PATH
|
|
||||||
security list-keychain -d user -s $KEYCHAIN_PATH
|
|
||||||
|
|
||||||
# Sign vendored binaries with hardened runtime and their specific entitlements
|
|
||||||
codesign --force --options runtime --entitlements crates-tauri/yaak-app/macos/entitlements.yaakprotoc.plist --sign "$APPLE_SIGNING_IDENTITY" crates-tauri/yaak-app/vendored/protoc/yaakprotoc || true
|
|
||||||
codesign --force --options runtime --entitlements crates-tauri/yaak-app/macos/entitlements.yaaknode.plist --sign "$APPLE_SIGNING_IDENTITY" crates-tauri/yaak-app/vendored/node/yaaknode || true
|
|
||||||
|
|
||||||
- uses: tauri-apps/tauri-action@v0
|
|
||||||
env:
|
|
||||||
YAAK_TARGET_ARCH: ${{ matrix.yaak_arch }}
|
|
||||||
|
|
||||||
ENABLE_CODE_SIGNING: ${{ secrets.APPLE_CERTIFICATE }}
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_PRIVATE_KEY }}
|
|
||||||
TAURI_SIGNING_PRIVATE_KEY_PASSWORD: ${{ secrets.TAURI_KEY_PASSWORD }}
|
|
||||||
|
|
||||||
# Apple signing stuff
|
|
||||||
APPLE_CERTIFICATE: ${{ matrix.os == 'macos' && secrets.APPLE_CERTIFICATE }}
|
|
||||||
APPLE_CERTIFICATE_PASSWORD: ${{ matrix.os == 'macos' && secrets.APPLE_CERTIFICATE_PASSWORD }}
|
|
||||||
APPLE_ID: ${{ matrix.os == 'macos' && secrets.APPLE_ID }}
|
|
||||||
APPLE_PASSWORD: ${{ matrix.os == 'macos' && secrets.APPLE_PASSWORD }}
|
|
||||||
APPLE_SIGNING_IDENTITY: ${{ matrix.os == 'macos' && secrets.APPLE_SIGNING_IDENTITY }}
|
|
||||||
APPLE_TEAM_ID: ${{ matrix.os == 'macos' && secrets.APPLE_TEAM_ID }}
|
|
||||||
|
|
||||||
# Windows signing stuff
|
|
||||||
AZURE_CLIENT_ID: ${{ matrix.os == 'windows' && secrets.AZURE_CLIENT_ID }}
|
|
||||||
AZURE_CLIENT_SECRET: ${{ matrix.os == 'windows' && secrets.AZURE_CLIENT_SECRET }}
|
|
||||||
AZURE_TENANT_ID: ${{ matrix.os == 'windows' && secrets.AZURE_TENANT_ID }}
|
|
||||||
with:
|
|
||||||
tagName: "v__VERSION__"
|
|
||||||
releaseName: "Release __VERSION__"
|
|
||||||
releaseBody: "[Changelog __VERSION__](https://yaak.app/blog/__VERSION__)"
|
|
||||||
releaseDraft: true
|
|
||||||
prerelease: true
|
|
||||||
args: "${{ matrix.args }} --config ./crates-tauri/yaak-app/tauri.release.conf.json"
|
|
||||||
|
|
||||||
# Build a per-machine NSIS installer for enterprise deployment (PDQ, SCCM, Intune)
|
|
||||||
- name: Build and upload machine-wide installer (Windows only)
|
|
||||||
if: matrix.os == 'windows'
|
|
||||||
shell: pwsh
|
|
||||||
env:
|
|
||||||
YAAK_TARGET_ARCH: ${{ matrix.yaak_arch }}
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
AZURE_CLIENT_ID: ${{ secrets.AZURE_CLIENT_ID }}
|
|
||||||
AZURE_CLIENT_SECRET: ${{ secrets.AZURE_CLIENT_SECRET }}
|
|
||||||
AZURE_TENANT_ID: ${{ secrets.AZURE_TENANT_ID }}
|
|
||||||
TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_PRIVATE_KEY }}
|
|
||||||
TAURI_SIGNING_PRIVATE_KEY_PASSWORD: ${{ secrets.TAURI_KEY_PASSWORD }}
|
|
||||||
run: |
|
|
||||||
Get-ChildItem -Recurse -Path target -File -Filter "*.exe.sig" | Remove-Item -Force
|
|
||||||
npx tauri bundle ${{ matrix.args }} --bundles nsis --config ./crates-tauri/yaak-app/tauri.release.conf.json --config '{"bundle":{"createUpdaterArtifacts":true,"windows":{"nsis":{"installMode":"perMachine"}}}}'
|
|
||||||
$setup = Get-ChildItem -Recurse -Path target -Filter "*setup*.exe" | Select-Object -First 1
|
|
||||||
$setupSig = "$($setup.FullName).sig"
|
|
||||||
$dest = $setup.FullName -replace '-setup\.exe$', '-setup-machine.exe'
|
|
||||||
$destSig = "$dest.sig"
|
|
||||||
Copy-Item $setup.FullName $dest
|
|
||||||
Copy-Item $setupSig $destSig
|
|
||||||
gh release upload "${{ github.ref_name }}" "$dest" --clobber
|
|
||||||
gh release upload "${{ github.ref_name }}" "$destSig" --clobber
|
|
||||||
218
.github/workflows/release-cli-npm.yml
vendored
218
.github/workflows/release-cli-npm.yml
vendored
@@ -1,218 +0,0 @@
|
|||||||
name: Release CLI to NPM
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags: [yaak-cli-*]
|
|
||||||
workflow_dispatch:
|
|
||||||
inputs:
|
|
||||||
version:
|
|
||||||
description: CLI version to publish (for example 0.4.0 or v0.4.0)
|
|
||||||
required: true
|
|
||||||
type: string
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
prepare-vendored-assets:
|
|
||||||
name: Prepare vendored plugin assets
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Setup Node
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: lts/*
|
|
||||||
|
|
||||||
- name: Install Rust stable
|
|
||||||
uses: dtolnay/rust-toolchain@stable
|
|
||||||
|
|
||||||
- name: Install dependencies
|
|
||||||
run: npm ci
|
|
||||||
|
|
||||||
- name: Build plugin assets
|
|
||||||
env:
|
|
||||||
SKIP_WASM_BUILD: "1"
|
|
||||||
run: |
|
|
||||||
npm run build
|
|
||||||
npm run vendor:vendor-plugins
|
|
||||||
|
|
||||||
- name: Upload vendored assets
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: vendored-assets
|
|
||||||
path: |
|
|
||||||
crates-tauri/yaak-app/vendored/plugin-runtime/index.cjs
|
|
||||||
crates-tauri/yaak-app/vendored/plugins
|
|
||||||
if-no-files-found: error
|
|
||||||
|
|
||||||
build-binaries:
|
|
||||||
name: Build ${{ matrix.pkg }}
|
|
||||||
needs: prepare-vendored-assets
|
|
||||||
runs-on: ${{ matrix.runner }}
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
- pkg: cli-darwin-arm64
|
|
||||||
runner: macos-latest
|
|
||||||
target: aarch64-apple-darwin
|
|
||||||
binary: yaak
|
|
||||||
- pkg: cli-darwin-x64
|
|
||||||
runner: macos-latest
|
|
||||||
target: x86_64-apple-darwin
|
|
||||||
binary: yaak
|
|
||||||
- pkg: cli-linux-arm64
|
|
||||||
runner: ubuntu-22.04-arm
|
|
||||||
target: aarch64-unknown-linux-gnu
|
|
||||||
binary: yaak
|
|
||||||
- pkg: cli-linux-x64
|
|
||||||
runner: ubuntu-22.04
|
|
||||||
target: x86_64-unknown-linux-gnu
|
|
||||||
binary: yaak
|
|
||||||
- pkg: cli-win32-arm64
|
|
||||||
runner: windows-latest
|
|
||||||
target: aarch64-pc-windows-msvc
|
|
||||||
binary: yaak.exe
|
|
||||||
- pkg: cli-win32-x64
|
|
||||||
runner: windows-latest
|
|
||||||
target: x86_64-pc-windows-msvc
|
|
||||||
binary: yaak.exe
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Install Rust stable
|
|
||||||
uses: dtolnay/rust-toolchain@stable
|
|
||||||
with:
|
|
||||||
targets: ${{ matrix.target }}
|
|
||||||
|
|
||||||
- name: Restore Rust cache
|
|
||||||
uses: Swatinem/rust-cache@v2
|
|
||||||
with:
|
|
||||||
shared-key: release-cli-npm
|
|
||||||
cache-on-failure: true
|
|
||||||
|
|
||||||
- name: Install Linux build dependencies
|
|
||||||
if: startsWith(matrix.runner, 'ubuntu')
|
|
||||||
run: |
|
|
||||||
sudo apt-get update
|
|
||||||
sudo apt-get install -y pkg-config libdbus-1-dev
|
|
||||||
|
|
||||||
- name: Download vendored assets
|
|
||||||
uses: actions/download-artifact@v4
|
|
||||||
with:
|
|
||||||
name: vendored-assets
|
|
||||||
path: crates-tauri/yaak-app/vendored
|
|
||||||
|
|
||||||
- name: Set CLI build version
|
|
||||||
shell: bash
|
|
||||||
env:
|
|
||||||
WORKFLOW_VERSION: ${{ inputs.version }}
|
|
||||||
run: |
|
|
||||||
set -euo pipefail
|
|
||||||
if [ "${{ github.event_name }}" = "workflow_dispatch" ]; then
|
|
||||||
VERSION="$WORKFLOW_VERSION"
|
|
||||||
else
|
|
||||||
VERSION="${GITHUB_REF_NAME#yaak-cli-}"
|
|
||||||
fi
|
|
||||||
VERSION="${VERSION#v}"
|
|
||||||
echo "Building yaak version: $VERSION"
|
|
||||||
echo "YAAK_CLI_VERSION=$VERSION" >> "$GITHUB_ENV"
|
|
||||||
|
|
||||||
- name: Build yaak
|
|
||||||
run: cargo build --locked --release -p yaak-cli --bin yaak --target ${{ matrix.target }}
|
|
||||||
|
|
||||||
- name: Stage binary artifact
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
set -euo pipefail
|
|
||||||
mkdir -p "npm/dist/${{ matrix.pkg }}"
|
|
||||||
cp "target/${{ matrix.target }}/release/${{ matrix.binary }}" "npm/dist/${{ matrix.pkg }}/${{ matrix.binary }}"
|
|
||||||
|
|
||||||
- name: Upload binary artifact
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: ${{ matrix.pkg }}
|
|
||||||
path: npm/dist/${{ matrix.pkg }}/${{ matrix.binary }}
|
|
||||||
if-no-files-found: error
|
|
||||||
|
|
||||||
publish-npm:
|
|
||||||
name: Publish @yaakapp/cli packages
|
|
||||||
needs: build-binaries
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
id-token: write
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Setup Node
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: lts/*
|
|
||||||
registry-url: https://registry.npmjs.org
|
|
||||||
|
|
||||||
- name: Download binary artifacts
|
|
||||||
uses: actions/download-artifact@v4
|
|
||||||
with:
|
|
||||||
pattern: cli-*
|
|
||||||
path: npm/dist
|
|
||||||
merge-multiple: false
|
|
||||||
|
|
||||||
- name: Prepare npm packages
|
|
||||||
shell: bash
|
|
||||||
env:
|
|
||||||
WORKFLOW_VERSION: ${{ inputs.version }}
|
|
||||||
run: |
|
|
||||||
set -euo pipefail
|
|
||||||
if [ "${{ github.event_name }}" = "workflow_dispatch" ]; then
|
|
||||||
VERSION="$WORKFLOW_VERSION"
|
|
||||||
else
|
|
||||||
VERSION="${GITHUB_REF_NAME#yaak-cli-}"
|
|
||||||
fi
|
|
||||||
VERSION="${VERSION#v}"
|
|
||||||
if [[ "$VERSION" == *-* ]]; then
|
|
||||||
PRERELEASE="${VERSION#*-}"
|
|
||||||
NPM_TAG="${PRERELEASE%%.*}"
|
|
||||||
else
|
|
||||||
NPM_TAG="latest"
|
|
||||||
fi
|
|
||||||
echo "Preparing CLI npm packages for version: $VERSION"
|
|
||||||
echo "Publishing with npm dist-tag: $NPM_TAG"
|
|
||||||
echo "NPM_TAG=$NPM_TAG" >> "$GITHUB_ENV"
|
|
||||||
YAAK_CLI_VERSION="$VERSION" node npm/prepare-publish.js
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/cli-darwin-arm64
|
|
||||||
run: npm publish --provenance --access public --tag "$NPM_TAG"
|
|
||||||
working-directory: npm/cli-darwin-arm64
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/cli-darwin-x64
|
|
||||||
run: npm publish --provenance --access public --tag "$NPM_TAG"
|
|
||||||
working-directory: npm/cli-darwin-x64
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/cli-linux-arm64
|
|
||||||
run: npm publish --provenance --access public --tag "$NPM_TAG"
|
|
||||||
working-directory: npm/cli-linux-arm64
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/cli-linux-x64
|
|
||||||
run: npm publish --provenance --access public --tag "$NPM_TAG"
|
|
||||||
working-directory: npm/cli-linux-x64
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/cli-win32-arm64
|
|
||||||
run: npm publish --provenance --access public --tag "$NPM_TAG"
|
|
||||||
working-directory: npm/cli-win32-arm64
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/cli-win32-x64
|
|
||||||
run: npm publish --provenance --access public --tag "$NPM_TAG"
|
|
||||||
working-directory: npm/cli-win32-x64
|
|
||||||
|
|
||||||
- name: Publish @yaakapp/cli
|
|
||||||
run: npm publish --provenance --access public --tag "$NPM_TAG"
|
|
||||||
working-directory: npm/cli
|
|
||||||
125
.github/workflows/release.yml
vendored
Normal file
125
.github/workflows/release.yml
vendored
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
name: Generate Artifacts
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags: [ v* ]
|
||||||
|
|
||||||
|
env:
|
||||||
|
YAAK_PLUGINS_DIR: checkout/plugins
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-artifacts:
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
name: Build
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- platform: 'macos-latest' # for Arm-based Macs (M1 and above).
|
||||||
|
args: '--target aarch64-apple-darwin'
|
||||||
|
yaak_arch: 'arm64'
|
||||||
|
- platform: 'macos-latest' # for Intel-based Macs.
|
||||||
|
args: '--target x86_64-apple-darwin'
|
||||||
|
yaak_arch: 'x64'
|
||||||
|
- platform: 'ubuntu-22.04'
|
||||||
|
args: ''
|
||||||
|
yaak_arch: 'x64'
|
||||||
|
- platform: 'windows-latest'
|
||||||
|
args: ''
|
||||||
|
yaak_arch: 'x64'
|
||||||
|
runs-on: ${{ matrix.platform }}
|
||||||
|
timeout-minutes: 40
|
||||||
|
steps:
|
||||||
|
- name: Checkout yaakapp/app
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 22
|
||||||
|
|
||||||
|
- name: install Rust stable
|
||||||
|
uses: dtolnay/rust-toolchain@stable
|
||||||
|
with:
|
||||||
|
# Those targets are only used on macos runners so it's in an `if` to slightly speed up windows and linux builds.
|
||||||
|
targets: ${{ matrix.platform == 'macos-latest' && 'aarch64-apple-darwin,x86_64-apple-darwin' || '' }}
|
||||||
|
|
||||||
|
- uses: actions/cache@v3
|
||||||
|
continue-on-error: false
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
~/.cargo/bin/
|
||||||
|
~/.cargo/registry/index/
|
||||||
|
~/.cargo/registry/cache/
|
||||||
|
~/.cargo/git/db/
|
||||||
|
src-tauri/target/
|
||||||
|
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
||||||
|
restore-keys: ${{ runner.os }}-cargo-
|
||||||
|
|
||||||
|
- name: install dependencies (ubuntu only)
|
||||||
|
if: matrix.platform == 'ubuntu-22.04' # This must match the platform value defined above.
|
||||||
|
run: |
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install -y libwebkit2gtk-4.1-dev libappindicator3-dev librsvg2-dev patchelf
|
||||||
|
|
||||||
|
- name: install dependencies (windows only)
|
||||||
|
if: matrix.platform == 'windows-latest'
|
||||||
|
run: cargo install --force trusted-signing-cli --version 0.5.0
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: |
|
||||||
|
npm ci
|
||||||
|
npm install @yaakapp/cli
|
||||||
|
|
||||||
|
- name: Install Protoc for plugin-runtime
|
||||||
|
uses: arduino/setup-protoc@v3
|
||||||
|
with:
|
||||||
|
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Run JS build
|
||||||
|
run: npm run build
|
||||||
|
|
||||||
|
- name: Run lint
|
||||||
|
run: npm run lint
|
||||||
|
|
||||||
|
- name: Checkout yaakapp/plugins
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
repository: yaakapp/plugins
|
||||||
|
path: ${{ env.YAAK_PLUGINS_DIR }}
|
||||||
|
|
||||||
|
- name: Set version
|
||||||
|
run: npm run replace-version
|
||||||
|
env:
|
||||||
|
YAAK_VERSION: ${{ github.ref_name }}
|
||||||
|
|
||||||
|
- uses: tauri-apps/tauri-action@v0
|
||||||
|
env:
|
||||||
|
YAAK_PLUGINS_DIR: ${{ env.YAAK_PLUGINS_DIR }}
|
||||||
|
YAAK_TARGET_ARCH: ${{ matrix.yaak_arch }}
|
||||||
|
|
||||||
|
ENABLE_CODE_SIGNING: ${{ secrets.APPLE_CERTIFICATE }}
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_PRIVATE_KEY }}
|
||||||
|
TAURI_SIGNING_PRIVATE_KEY_PASSWORD: ${{ secrets.TAURI_KEY_PASSWORD }}
|
||||||
|
|
||||||
|
# Apple signing stuff
|
||||||
|
APPLE_CERTIFICATE: ${{ matrix.platform == 'macos-latest' && secrets.APPLE_CERTIFICATE }}
|
||||||
|
APPLE_CERTIFICATE_PASSWORD: ${{ matrix.platform == 'macos-latest' && secrets.APPLE_CERTIFICATE_PASSWORD }}
|
||||||
|
APPLE_ID: ${{ matrix.platform == 'macos-latest' && secrets.APPLE_ID }}
|
||||||
|
APPLE_PASSWORD: ${{ matrix.platform == 'macos-latest' && secrets.APPLE_PASSWORD }}
|
||||||
|
APPLE_SIGNING_IDENTITY: ${{ matrix.platform == 'macos-latest' && secrets.APPLE_SIGNING_IDENTITY }}
|
||||||
|
APPLE_TEAM_ID: ${{ matrix.platform == 'macos-latest' && secrets.APPLE_TEAM_ID }}
|
||||||
|
|
||||||
|
# Windows signing stuff
|
||||||
|
AZURE_CLIENT_ID: ${{ matrix.platform == 'windows-latest' && secrets.AZURE_CLIENT_ID }}
|
||||||
|
AZURE_CLIENT_SECRET: ${{ matrix.platform == 'windows-latest' && secrets.AZURE_CLIENT_SECRET }}
|
||||||
|
AZURE_TENANT_ID: ${{ matrix.platform == 'windows-latest' && secrets.AZURE_TENANT_ID }}
|
||||||
|
with:
|
||||||
|
tagName: 'v__VERSION__'
|
||||||
|
releaseName: 'Release __VERSION__'
|
||||||
|
releaseBody: '[Changelog __VERSION__](https://yaak.app/blog/__VERSION__)'
|
||||||
|
releaseDraft: true
|
||||||
|
prerelease: false
|
||||||
|
args: ${{ matrix.args }}
|
||||||
44
.github/workflows/sponsors.yml
vendored
44
.github/workflows/sponsors.yml
vendored
@@ -1,44 +0,0 @@
|
|||||||
name: Generate Sponsors README
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
schedule:
|
|
||||||
- cron: 30 15 * * 0-6
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
jobs:
|
|
||||||
deploy:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout 🛎️
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Generate Sponsors
|
|
||||||
uses: JamesIves/github-sponsors-readme-action@v1
|
|
||||||
with:
|
|
||||||
token: ${{ secrets.SPONSORS_PAT }}
|
|
||||||
file: 'README.md'
|
|
||||||
maximum: 1999
|
|
||||||
template: '<a href="https://github.com/{{{ login }}}"><img src="{{{ avatarUrl }}}" width="50px" alt="User avatar: {{{ login }}}" /></a> '
|
|
||||||
active-only: false
|
|
||||||
include-private: true
|
|
||||||
marker: 'sponsors-base'
|
|
||||||
|
|
||||||
- name: Generate Sponsors
|
|
||||||
uses: JamesIves/github-sponsors-readme-action@v1
|
|
||||||
with:
|
|
||||||
token: ${{ secrets.SPONSORS_PAT }}
|
|
||||||
file: 'README.md'
|
|
||||||
minimum: 2000
|
|
||||||
template: '<a href="https://github.com/{{{ login }}}"><img src="{{{ avatarUrl }}}" width="80px" alt="User avatar: {{{ login }}}" /></a> '
|
|
||||||
active-only: false
|
|
||||||
include-private: true
|
|
||||||
marker: 'sponsors-premium'
|
|
||||||
|
|
||||||
# ⚠️ Note: You can use any deployment step here to automatically push the README
|
|
||||||
# changes back to your branch.
|
|
||||||
- name: Commit Changes
|
|
||||||
uses: JamesIves/github-pages-deploy-action@v4
|
|
||||||
with:
|
|
||||||
branch: main
|
|
||||||
force: false
|
|
||||||
folder: '.'
|
|
||||||
23
.gitignore
vendored
23
.gitignore
vendored
@@ -15,8 +15,6 @@ dist-ssr
|
|||||||
# Editor directories and files
|
# Editor directories and files
|
||||||
.vscode/*
|
.vscode/*
|
||||||
!.vscode/extensions.json
|
!.vscode/extensions.json
|
||||||
!.vscode/settings.json
|
|
||||||
!.vscode/launch.json
|
|
||||||
.idea
|
.idea
|
||||||
.DS_Store
|
.DS_Store
|
||||||
*.suo
|
*.suo
|
||||||
@@ -25,7 +23,6 @@ dist-ssr
|
|||||||
*.sln
|
*.sln
|
||||||
*.sw?
|
*.sw?
|
||||||
.eslintcache
|
.eslintcache
|
||||||
out
|
|
||||||
|
|
||||||
*.sqlite
|
*.sqlite
|
||||||
*.sqlite-*
|
*.sqlite-*
|
||||||
@@ -34,23 +31,3 @@ out
|
|||||||
|
|
||||||
.tmp
|
.tmp
|
||||||
tmp
|
tmp
|
||||||
.zed
|
|
||||||
codebook.toml
|
|
||||||
target
|
|
||||||
|
|
||||||
# Per-worktree Tauri config (generated by post-checkout hook)
|
|
||||||
crates-tauri/yaak-app/tauri.worktree.conf.json
|
|
||||||
|
|
||||||
# Tauri auto-generated permission files
|
|
||||||
**/permissions/autogenerated
|
|
||||||
**/permissions/schemas
|
|
||||||
|
|
||||||
# Flatpak build artifacts
|
|
||||||
flatpak-repo/
|
|
||||||
.flatpak-builder/
|
|
||||||
flatpak/flatpak-builder-tools/
|
|
||||||
flatpak/cargo-sources.json
|
|
||||||
flatpak/node-sources.json
|
|
||||||
|
|
||||||
# Local Codex desktop env state
|
|
||||||
.codex/environments/environment.toml
|
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
node scripts/git-hooks/post-checkout.mjs "$@"
|
|
||||||
4
.prettierignore
Normal file
4
.prettierignore
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
node_modules/
|
||||||
|
dist/
|
||||||
|
out/
|
||||||
|
.prettierrc.cjs
|
||||||
8
.prettierrc.js
Normal file
8
.prettierrc.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
export default {
|
||||||
|
"trailingComma": "all",
|
||||||
|
"tabWidth": 2,
|
||||||
|
"semi": true,
|
||||||
|
"singleQuote": true,
|
||||||
|
"printWidth": 100,
|
||||||
|
"bracketSpacing": true
|
||||||
|
}
|
||||||
3
.vscode/extensions.json
vendored
3
.vscode/extensions.json
vendored
@@ -1,3 +0,0 @@
|
|||||||
{
|
|
||||||
"recommendations": ["biomejs.biome", "rust-lang.rust-analyzer", "bradlc.vscode-tailwindcss"]
|
|
||||||
}
|
|
||||||
26
.vscode/launch.json
vendored
26
.vscode/launch.json
vendored
@@ -1,26 +0,0 @@
|
|||||||
{
|
|
||||||
"version": "0.2.0",
|
|
||||||
"configurations": [
|
|
||||||
{
|
|
||||||
"type": "node",
|
|
||||||
"request": "launch",
|
|
||||||
"name": "Dev App",
|
|
||||||
"runtimeExecutable": "npm",
|
|
||||||
"runtimeArgs": ["run", "start"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "node",
|
|
||||||
"request": "launch",
|
|
||||||
"name": "Build App",
|
|
||||||
"runtimeExecutable": "npm",
|
|
||||||
"runtimeArgs": ["run", "start"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "node",
|
|
||||||
"request": "launch",
|
|
||||||
"name": "Bootstrap",
|
|
||||||
"runtimeExecutable": "npm",
|
|
||||||
"runtimeArgs": ["run", "bootstrap"]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
6
.vscode/settings.json
vendored
6
.vscode/settings.json
vendored
@@ -1,6 +0,0 @@
|
|||||||
{
|
|
||||||
"editor.defaultFormatter": "biomejs.biome",
|
|
||||||
"editor.formatOnSave": true,
|
|
||||||
"biome.enabled": true,
|
|
||||||
"biome.lint.format.enable": true
|
|
||||||
}
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
# Contributing to Yaak
|
|
||||||
|
|
||||||
Yaak accepts community pull requests for:
|
|
||||||
|
|
||||||
- Bug fixes
|
|
||||||
- Small-scope improvements directly tied to existing behavior
|
|
||||||
|
|
||||||
Pull requests that introduce broad new features, major redesigns, or large refactors are out of scope unless explicitly approved first.
|
|
||||||
|
|
||||||
## Approval for Non-Bugfix Changes
|
|
||||||
|
|
||||||
If your PR is not a bug fix or small-scope improvement, include a link to the approved [feedback item](https://yaak.app/feedback) where contribution approval was explicitly stated.
|
|
||||||
|
|
||||||
## Development Setup
|
|
||||||
|
|
||||||
For local setup and development workflows, see [`DEVELOPMENT.md`](DEVELOPMENT.md).
|
|
||||||
74
Cargo.toml
74
Cargo.toml
@@ -1,74 +0,0 @@
|
|||||||
[workspace]
|
|
||||||
resolver = "2"
|
|
||||||
members = [
|
|
||||||
"crates/yaak",
|
|
||||||
# Shared crates (no Tauri dependency)
|
|
||||||
"crates/yaak-core",
|
|
||||||
"crates/yaak-common",
|
|
||||||
"crates/yaak-crypto",
|
|
||||||
"crates/yaak-git",
|
|
||||||
"crates/yaak-grpc",
|
|
||||||
"crates/yaak-http",
|
|
||||||
"crates/yaak-models",
|
|
||||||
"crates/yaak-plugins",
|
|
||||||
"crates/yaak-sse",
|
|
||||||
"crates/yaak-sync",
|
|
||||||
"crates/yaak-templates",
|
|
||||||
"crates/yaak-tls",
|
|
||||||
"crates/yaak-ws",
|
|
||||||
"crates/yaak-api",
|
|
||||||
# CLI crates
|
|
||||||
"crates-cli/yaak-cli",
|
|
||||||
# Tauri-specific crates
|
|
||||||
"crates-tauri/yaak-app",
|
|
||||||
"crates-tauri/yaak-fonts",
|
|
||||||
"crates-tauri/yaak-license",
|
|
||||||
"crates-tauri/yaak-mac-window",
|
|
||||||
"crates-tauri/yaak-tauri-utils",
|
|
||||||
]
|
|
||||||
|
|
||||||
[workspace.dependencies]
|
|
||||||
chrono = "0.4.42"
|
|
||||||
hex = "0.4.3"
|
|
||||||
keyring = "3.6.3"
|
|
||||||
log = "0.4.29"
|
|
||||||
reqwest = "0.12.20"
|
|
||||||
rustls = { version = "0.23.34", default-features = false }
|
|
||||||
rustls-platform-verifier = "0.6.2"
|
|
||||||
schemars = { version = "0.8.22", features = ["chrono"] }
|
|
||||||
serde = "1.0.228"
|
|
||||||
serde_json = "1.0.145"
|
|
||||||
sha2 = "0.10.9"
|
|
||||||
tauri = "2.9.5"
|
|
||||||
tauri-plugin = "2.5.2"
|
|
||||||
tauri-plugin-dialog = "2.4.2"
|
|
||||||
tauri-plugin-shell = "2.3.3"
|
|
||||||
thiserror = "2.0.17"
|
|
||||||
tokio = "1.48.0"
|
|
||||||
ts-rs = "11.1.0"
|
|
||||||
|
|
||||||
# Internal crates - shared
|
|
||||||
yaak-core = { path = "crates/yaak-core" }
|
|
||||||
yaak = { path = "crates/yaak" }
|
|
||||||
yaak-common = { path = "crates/yaak-common" }
|
|
||||||
yaak-crypto = { path = "crates/yaak-crypto" }
|
|
||||||
yaak-git = { path = "crates/yaak-git" }
|
|
||||||
yaak-grpc = { path = "crates/yaak-grpc" }
|
|
||||||
yaak-http = { path = "crates/yaak-http" }
|
|
||||||
yaak-models = { path = "crates/yaak-models" }
|
|
||||||
yaak-plugins = { path = "crates/yaak-plugins" }
|
|
||||||
yaak-sse = { path = "crates/yaak-sse" }
|
|
||||||
yaak-sync = { path = "crates/yaak-sync" }
|
|
||||||
yaak-templates = { path = "crates/yaak-templates" }
|
|
||||||
yaak-tls = { path = "crates/yaak-tls" }
|
|
||||||
yaak-ws = { path = "crates/yaak-ws" }
|
|
||||||
yaak-api = { path = "crates/yaak-api" }
|
|
||||||
|
|
||||||
# Internal crates - Tauri-specific
|
|
||||||
yaak-fonts = { path = "crates-tauri/yaak-fonts" }
|
|
||||||
yaak-license = { path = "crates-tauri/yaak-license" }
|
|
||||||
yaak-mac-window = { path = "crates-tauri/yaak-mac-window" }
|
|
||||||
yaak-tauri-utils = { path = "crates-tauri/yaak-tauri-utils" }
|
|
||||||
|
|
||||||
[profile.release]
|
|
||||||
strip = false
|
|
||||||
@@ -34,6 +34,8 @@ Run the `bootstrap` command to do some initial setup:
|
|||||||
npm run bootstrap
|
npm run bootstrap
|
||||||
```
|
```
|
||||||
|
|
||||||
|
_NOTE: Run with `YAAK_PLUGINS_DIR=<Path to yaakapp/plugins>` to re-build bundled plugins_
|
||||||
|
|
||||||
## Run the App
|
## Run the App
|
||||||
|
|
||||||
After bootstrapping, start the app in development mode:
|
After bootstrapping, start the app in development mode:
|
||||||
@@ -42,47 +44,26 @@ After bootstrapping, start the app in development mode:
|
|||||||
npm start
|
npm start
|
||||||
```
|
```
|
||||||
|
|
||||||
|
_NOTE: If working on bundled plugins, run with `YAAK_PLUGINS_DIR=<Path to yaakapp/plugins>`_
|
||||||
|
|
||||||
## SQLite Migrations
|
## SQLite Migrations
|
||||||
|
|
||||||
New migrations can be created from the `src-tauri/` directory:
|
New migrations can be created from the `src-tauri/` directory:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
npm run migration
|
cd src-tauri
|
||||||
|
sqlx migrate add migration-name
|
||||||
```
|
```
|
||||||
|
|
||||||
Rerun the app to apply the migrations.
|
Run the app to apply the migrations.
|
||||||
|
|
||||||
_Note: For safety, development builds use a separate database location from production builds._
|
If nothing happens, try `cargo clean` and run the app again.
|
||||||
|
|
||||||
## Lezer Grammar Generation
|
_Note: Development builds use a separate database location from production builds._
|
||||||
|
|
||||||
|
## Lezer Grammer Generation
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
# Example
|
# Example
|
||||||
lezer-generator components/core/Editor/<LANG>/<LANG>.grammar > components/core/Editor/<LANG>/<LANG>.ts
|
lezer-generator components/core/Editor/<LANG>/<LANG>.grammar > components/core/Editor/<LANG>/<LANG>.ts
|
||||||
```
|
```
|
||||||
|
|
||||||
## Linting & Formatting
|
|
||||||
|
|
||||||
This repo uses Biome for linting and formatting (replacing ESLint + Prettier).
|
|
||||||
|
|
||||||
- Lint the entire repo:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm run lint
|
|
||||||
```
|
|
||||||
|
|
||||||
- Auto-fix lint issues where possible:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm run lint:fix
|
|
||||||
```
|
|
||||||
|
|
||||||
- Format code:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm run format
|
|
||||||
```
|
|
||||||
|
|
||||||
Notes:
|
|
||||||
- Many workspace packages also expose the same scripts (`lint`, `lint:fix`, and `format`).
|
|
||||||
- TypeScript type-checking still runs separately via `tsc --noEmit` in relevant packages.
|
|
||||||
|
|||||||
88
README.md
88
README.md
@@ -1,72 +1,34 @@
|
|||||||
<p align="center">
|
# Yaak API Client
|
||||||
<a href="https://github.com/JamesIves/github-sponsors-readme-action">
|
|
||||||
<img width="200px" src="https://github.com/mountain-loop/yaak/raw/main/crates-tauri/yaak-app/icons/icon.png">
|
|
||||||
</a>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<h1 align="center">
|
Yaak is a desktop API client for interacting with REST, GraphQL, Server Sent Events (SSE), WebSocket, and gRPC
|
||||||
💫 Yaak ➟ Desktop API Client 💫
|
APIs. It's built using [Tauri](https://tauri.app), Rust, and ReactJS.
|
||||||
</h1>
|
|
||||||
|
|
||||||
<p align="center">
|
|
||||||
A fast, privacy-first API client for REST, GraphQL, SSE, WebSocket, and gRPC – built with Tauri, Rust, and React.
|
|
||||||
</p>
|
|
||||||
<p align="center">
|
|
||||||
Development is funded by community-purchased <a href="https://yaak.app/pricing">licenses</a>. You can also <a href="https://github.com/sponsors/gschier">become a sponsor</a> to have your logo appear below. 💖
|
|
||||||
</p>
|
|
||||||
<br>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<p align="center">
|
|
||||||
<!-- 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 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> <a href="https://github.com/flashblaze"><img src="https://github.com/flashblaze.png" width="50px" alt="User avatar: flashblaze" /></a> <a href="https://github.com/Frostist"><img src="https://github.com/Frostist.png" width="50px" alt="User avatar: Frostist" /></a> <!-- sponsors-base -->
|
|
||||||
</p>
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
|
|
||||||
## Features
|
|
||||||
|
|
||||||
Yaak is an offline-first API client designed to stay out of your way while giving you everything you need when you need it.
|
|
||||||
Built with [Tauri](https://tauri.app), Rust, and React, it’s fast, lightweight, and private. No telemetry, no VC funding, and no cloud lock-in.
|
|
||||||
|
|
||||||
|
|
||||||
### 🌐 Work with any API
|
|
||||||
|
|
||||||
- Import collections from Postman, Insomnia, OpenAPI, Swagger, or Curl.
|
|
||||||
- Send requests via REST, GraphQL, gRPC, WebSocket, or Server-Sent Events.
|
|
||||||
- Filter and inspect responses with JSONPath or XPath.
|
|
||||||
|
|
||||||
### 🔐 Stay secure
|
|
||||||
- Use OAuth 2.0, JWT, Basic Auth, or custom plugins for authentication.
|
|
||||||
- Secure sensitive values with encrypted secrets.
|
|
||||||
- Store secrets in your OS keychain.
|
|
||||||
|
|
||||||
### ☁️ Organize & collaborate
|
|
||||||
- Group requests into workspaces and nested folders.
|
|
||||||
- Use environment variables to switch between dev, staging, and prod.
|
|
||||||
- Mirror workspaces to your filesystem for versioning in Git or syncing with Dropbox.
|
|
||||||
|
|
||||||
### 🧩 Extend & customize
|
|
||||||
- Insert dynamic values like UUIDs or timestamps with template tags.
|
|
||||||
- Pick from built-in themes or build your own.
|
|
||||||
- Create plugins to extend authentication, template tags, or the UI.
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## Contribution Policy
|
## Contribution Policy
|
||||||
|
|
||||||
> [!IMPORTANT]
|
Yaak is open source, but only accepting contributions for bug fixes. To get started,
|
||||||
> Community PRs are currently limited to bug fixes and small-scope improvements.
|
visit [`DEVELOPMENT.md`](DEVELOPMENT.md) for tips on setting up your environment.
|
||||||
> If your PR is out of scope, link an approved feedback item from [yaak.app/feedback](https://yaak.app/feedback).
|
|
||||||
> See [`CONTRIBUTING.md`](CONTRIBUTING.md) for policy details and [`DEVELOPMENT.md`](DEVELOPMENT.md) for local setup.
|
## Feature Overview
|
||||||
|
|
||||||
|
- 🪂 Import data from Postman, Insomnia, OpenAPI, Swagger, or Curl.<br/>
|
||||||
|
- 📤 Send requests via REST, GraphQL, Server Sent Events (SSE), WebSockets, or gRPC.<br/>
|
||||||
|
- 🔐 Automatically authorize requests with OAuth 2.0, JWT tokens, Basic Auth, and more.<br/>
|
||||||
|
- 🔎 Filter response bodies using JSONPath or XPath queries.<br/>
|
||||||
|
- ⛓️ Chain together multiple requests to dynamically reference values.<br/>
|
||||||
|
- 📂 Organize requests into workspaces and nested folders.<br/>
|
||||||
|
- 🧮 Use environment variables to easily switch between Prod and Dev.<br/>
|
||||||
|
- 🛡️ Secure arbitrary text values with end-to-end encryption<br/>
|
||||||
|
- 🏷️ Send dynamic values like UUIDs or timestamps using template tags.<br/>
|
||||||
|
- 🎨 Choose from many of the included themes, or make your own.<br/>
|
||||||
|
- 💽 Mirror workspace data to a directory for integration with Git or Dropbox.<br/>
|
||||||
|
- 📜 View response history for each request.<br/>
|
||||||
|
- 🔌 Create your own plugins for authentication, template tags, and more!<br/>
|
||||||
|
- 🛜 Configure a proxy to access firewall-blocked APIs
|
||||||
|
|
||||||
## Useful Resources
|
## Useful Resources
|
||||||
|
|
||||||
- [Feedback and Bug Reports](https://feedback.yaak.app)
|
- [Feedback and Bug Reports](https://feedback.yaak.app)
|
||||||
- [Documentation](https://yaak.app/docs)
|
- [Documentation](https://feedback.yaak.app/help)
|
||||||
- [Yaak vs Postman](https://yaak.app/alternatives/postman)
|
- [Yaak vs Postman](https://yaak.app/blog/postman-alternative)
|
||||||
- [Yaak vs Bruno](https://yaak.app/alternatives/bruno)
|
|
||||||
- [Yaak vs Insomnia](https://yaak.app/alternatives/insomnia)
|
|
||||||
|
|||||||
54
biome.json
54
biome.json
@@ -1,54 +0,0 @@
|
|||||||
{
|
|
||||||
"$schema": "https://biomejs.dev/schemas/2.3.11/schema.json",
|
|
||||||
"linter": {
|
|
||||||
"enabled": true,
|
|
||||||
"rules": {
|
|
||||||
"recommended": true,
|
|
||||||
"a11y": {
|
|
||||||
"useKeyWithClickEvents": "off"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"formatter": {
|
|
||||||
"enabled": true,
|
|
||||||
"indentStyle": "space",
|
|
||||||
"indentWidth": 2,
|
|
||||||
"lineWidth": 100,
|
|
||||||
"bracketSpacing": true
|
|
||||||
},
|
|
||||||
"css": {
|
|
||||||
"parser": {
|
|
||||||
"tailwindDirectives": true
|
|
||||||
},
|
|
||||||
"linter": {
|
|
||||||
"enabled": false
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"javascript": {
|
|
||||||
"formatter": {
|
|
||||||
"quoteStyle": "single",
|
|
||||||
"jsxQuoteStyle": "double",
|
|
||||||
"trailingCommas": "all",
|
|
||||||
"semicolons": "always"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"files": {
|
|
||||||
"includes": [
|
|
||||||
"**",
|
|
||||||
"!**/node_modules",
|
|
||||||
"!**/dist",
|
|
||||||
"!**/build",
|
|
||||||
"!target",
|
|
||||||
"!scripts",
|
|
||||||
"!crates",
|
|
||||||
"!crates-tauri",
|
|
||||||
"!src-web/tailwind.config.cjs",
|
|
||||||
"!src-web/postcss.config.cjs",
|
|
||||||
"!src-web/vite.config.ts",
|
|
||||||
"!src-web/routeTree.gen.ts",
|
|
||||||
"!packages/plugin-runtime-types/lib",
|
|
||||||
"!**/bindings",
|
|
||||||
"!flatpak"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "yaak-cli"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2024"
|
|
||||||
publish = false
|
|
||||||
|
|
||||||
[[bin]]
|
|
||||||
name = "yaak"
|
|
||||||
path = "src/main.rs"
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
base64 = "0.22"
|
|
||||||
clap = { version = "4", features = ["derive"] }
|
|
||||||
console = "0.15"
|
|
||||||
dirs = "6"
|
|
||||||
env_logger = "0.11"
|
|
||||||
futures = "0.3"
|
|
||||||
hex = { workspace = true }
|
|
||||||
include_dir = "0.7"
|
|
||||||
keyring = { workspace = true, features = ["apple-native", "windows-native", "sync-secret-service"] }
|
|
||||||
log = { workspace = true }
|
|
||||||
rand = "0.8"
|
|
||||||
reqwest = { workspace = true }
|
|
||||||
rolldown = "0.1.0"
|
|
||||||
oxc_resolver = "=11.10.0"
|
|
||||||
schemars = { workspace = true }
|
|
||||||
serde = { workspace = true }
|
|
||||||
serde_json = { workspace = true }
|
|
||||||
sha2 = { workspace = true }
|
|
||||||
tokio = { workspace = true, features = ["rt-multi-thread", "macros", "io-util", "net", "signal", "time"] }
|
|
||||||
walkdir = "2"
|
|
||||||
webbrowser = "1"
|
|
||||||
zip = "4"
|
|
||||||
yaak = { workspace = true }
|
|
||||||
yaak-crypto = { workspace = true }
|
|
||||||
yaak-http = { workspace = true }
|
|
||||||
yaak-models = { workspace = true }
|
|
||||||
yaak-plugins = { workspace = true }
|
|
||||||
yaak-templates = { workspace = true }
|
|
||||||
|
|
||||||
[dev-dependencies]
|
|
||||||
assert_cmd = "2"
|
|
||||||
predicates = "3"
|
|
||||||
tempfile = "3"
|
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
# Yaak CLI
|
|
||||||
|
|
||||||
The `yaak` CLI for publishing plugins and creating/updating/sending requests.
|
|
||||||
|
|
||||||
## Installation
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm install @yaakapp/cli
|
|
||||||
```
|
|
||||||
|
|
||||||
## Agentic Workflows
|
|
||||||
|
|
||||||
The `yaak` CLI is primarily meant to be used by AI agents, and has the following features:
|
|
||||||
|
|
||||||
- `schema` subcommands to get the JSON Schema for any model (eg. `yaak request schema http`)
|
|
||||||
- `--json '{...}'` input format to create and update data
|
|
||||||
- `--verbose` mode for extracting debug info while sending requests
|
|
||||||
- The ability to send entire workspaces and folders (Supports `--parallel` and `--fail-fast`)
|
|
||||||
|
|
||||||
### Example Prompts
|
|
||||||
|
|
||||||
Use the `yaak` CLI with agents like Claude or Codex to do useful things for you.
|
|
||||||
|
|
||||||
Here are some example prompts:
|
|
||||||
|
|
||||||
```text
|
|
||||||
Scan my API routes and create a workspace (using yaak cli) with
|
|
||||||
all the requests needed for me to do manual testing?
|
|
||||||
```
|
|
||||||
|
|
||||||
```text
|
|
||||||
Send all the GraphQL requests in my workspace
|
|
||||||
```
|
|
||||||
|
|
||||||
## Description
|
|
||||||
|
|
||||||
Here's the current print of `yaak --help`
|
|
||||||
|
|
||||||
```text
|
|
||||||
Yaak CLI - API client from the command line
|
|
||||||
|
|
||||||
Usage: yaak [OPTIONS] <COMMAND>
|
|
||||||
|
|
||||||
Commands:
|
|
||||||
auth Authentication commands
|
|
||||||
plugin Plugin development and publishing commands
|
|
||||||
send Send a request, folder, or workspace by ID
|
|
||||||
workspace Workspace commands
|
|
||||||
request Request commands
|
|
||||||
folder Folder commands
|
|
||||||
environment Environment commands
|
|
||||||
|
|
||||||
Options:
|
|
||||||
--data-dir <DATA_DIR> Use a custom data directory
|
|
||||||
-e, --environment <ENVIRONMENT> Environment ID to use for variable substitution
|
|
||||||
-v, --verbose Enable verbose send output (events and streamed response body)
|
|
||||||
--log [<LEVEL>] Enable CLI logging; optionally set level (error|warn|info|debug|trace) [possible values: error, warn, info, debug, trace]
|
|
||||||
-h, --help Print help
|
|
||||||
-V, --version Print version
|
|
||||||
|
|
||||||
Agent Hints:
|
|
||||||
- Template variable syntax is ${[ my_var ]}, not {{ ... }}
|
|
||||||
- Template function syntax is ${[ namespace.my_func(a='aaa',b='bbb') ]}
|
|
||||||
- View JSONSchema for models before creating or updating (eg. `yaak request schema http`)
|
|
||||||
- Deletion requires confirmation (--yes for non-interactive environments)
|
|
||||||
```
|
|
||||||
@@ -1,435 +0,0 @@
|
|||||||
use clap::{Args, Parser, Subcommand, ValueEnum};
|
|
||||||
use std::path::PathBuf;
|
|
||||||
|
|
||||||
#[derive(Parser)]
|
|
||||||
#[command(name = "yaak")]
|
|
||||||
#[command(about = "Yaak CLI - API client from the command line")]
|
|
||||||
#[command(version = crate::version::cli_version())]
|
|
||||||
#[command(disable_help_subcommand = true)]
|
|
||||||
#[command(after_help = r#"Agent Hints:
|
|
||||||
- Template variable syntax is ${[ my_var ]}, not {{ ... }}
|
|
||||||
- Template function syntax is ${[ namespace.my_func(a='aaa',b='bbb') ]}
|
|
||||||
- View JSONSchema for models before creating or updating (eg. `yaak request schema http`)
|
|
||||||
- Deletion requires confirmation (--yes for non-interactive environments)
|
|
||||||
"#)]
|
|
||||||
pub struct Cli {
|
|
||||||
/// Use a custom data directory
|
|
||||||
#[arg(long, global = true)]
|
|
||||||
pub data_dir: Option<PathBuf>,
|
|
||||||
|
|
||||||
/// Environment ID to use for variable substitution
|
|
||||||
#[arg(long, short, global = true)]
|
|
||||||
pub environment: Option<String>,
|
|
||||||
|
|
||||||
/// Enable verbose send output (events and streamed response body)
|
|
||||||
#[arg(long, short, global = true)]
|
|
||||||
pub verbose: bool,
|
|
||||||
|
|
||||||
/// Enable CLI logging; optionally set level (error|warn|info|debug|trace)
|
|
||||||
#[arg(long, global = true, value_name = "LEVEL", num_args = 0..=1, ignore_case = true)]
|
|
||||||
pub log: Option<Option<LogLevel>>,
|
|
||||||
|
|
||||||
#[command(subcommand)]
|
|
||||||
pub command: Commands,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand)]
|
|
||||||
pub enum Commands {
|
|
||||||
/// Authentication commands
|
|
||||||
Auth(AuthArgs),
|
|
||||||
|
|
||||||
/// Plugin development and publishing commands
|
|
||||||
Plugin(PluginArgs),
|
|
||||||
|
|
||||||
#[command(hide = true)]
|
|
||||||
Build(PluginPathArg),
|
|
||||||
|
|
||||||
#[command(hide = true)]
|
|
||||||
Dev(PluginPathArg),
|
|
||||||
|
|
||||||
/// Send a request, folder, or workspace by ID
|
|
||||||
Send(SendArgs),
|
|
||||||
|
|
||||||
/// Workspace commands
|
|
||||||
Workspace(WorkspaceArgs),
|
|
||||||
|
|
||||||
/// Request commands
|
|
||||||
Request(RequestArgs),
|
|
||||||
|
|
||||||
/// Folder commands
|
|
||||||
Folder(FolderArgs),
|
|
||||||
|
|
||||||
/// Environment commands
|
|
||||||
Environment(EnvironmentArgs),
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args)]
|
|
||||||
pub struct SendArgs {
|
|
||||||
/// Request, folder, or workspace ID
|
|
||||||
pub id: String,
|
|
||||||
|
|
||||||
/// Execute requests in parallel
|
|
||||||
#[arg(long)]
|
|
||||||
pub parallel: bool,
|
|
||||||
|
|
||||||
/// Stop on first request failure when sending folders/workspaces
|
|
||||||
#[arg(long, conflicts_with = "parallel")]
|
|
||||||
pub fail_fast: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args)]
|
|
||||||
#[command(disable_help_subcommand = true)]
|
|
||||||
pub struct WorkspaceArgs {
|
|
||||||
#[command(subcommand)]
|
|
||||||
pub command: WorkspaceCommands,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand)]
|
|
||||||
pub enum WorkspaceCommands {
|
|
||||||
/// List all workspaces
|
|
||||||
List,
|
|
||||||
|
|
||||||
/// Output JSON schema for workspace create/update payloads
|
|
||||||
Schema {
|
|
||||||
/// Pretty-print schema JSON output
|
|
||||||
#[arg(long)]
|
|
||||||
pretty: bool,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Show a workspace as JSON
|
|
||||||
Show {
|
|
||||||
/// Workspace ID
|
|
||||||
workspace_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Create a workspace
|
|
||||||
Create {
|
|
||||||
/// Workspace name
|
|
||||||
#[arg(short, long)]
|
|
||||||
name: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload
|
|
||||||
#[arg(long, conflicts_with = "json_input")]
|
|
||||||
json: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload shorthand
|
|
||||||
#[arg(value_name = "JSON", conflicts_with = "json")]
|
|
||||||
json_input: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Update a workspace
|
|
||||||
Update {
|
|
||||||
/// JSON payload
|
|
||||||
#[arg(long, conflicts_with = "json_input")]
|
|
||||||
json: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload shorthand
|
|
||||||
#[arg(value_name = "JSON", conflicts_with = "json")]
|
|
||||||
json_input: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Delete a workspace
|
|
||||||
Delete {
|
|
||||||
/// Workspace ID
|
|
||||||
workspace_id: String,
|
|
||||||
|
|
||||||
/// Skip confirmation prompt
|
|
||||||
#[arg(short, long)]
|
|
||||||
yes: bool,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args)]
|
|
||||||
#[command(disable_help_subcommand = true)]
|
|
||||||
pub struct RequestArgs {
|
|
||||||
#[command(subcommand)]
|
|
||||||
pub command: RequestCommands,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand)]
|
|
||||||
pub enum RequestCommands {
|
|
||||||
/// List requests in a workspace
|
|
||||||
List {
|
|
||||||
/// Workspace ID
|
|
||||||
workspace_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Show a request as JSON
|
|
||||||
Show {
|
|
||||||
/// Request ID
|
|
||||||
request_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Send a request by ID
|
|
||||||
Send {
|
|
||||||
/// Request ID
|
|
||||||
request_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Output JSON schema for request create/update payloads
|
|
||||||
Schema {
|
|
||||||
#[arg(value_enum)]
|
|
||||||
request_type: RequestSchemaType,
|
|
||||||
|
|
||||||
/// Pretty-print schema JSON output
|
|
||||||
#[arg(long)]
|
|
||||||
pretty: bool,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Create a new HTTP request
|
|
||||||
Create {
|
|
||||||
/// Workspace ID (or positional JSON payload shorthand)
|
|
||||||
workspace_id: Option<String>,
|
|
||||||
|
|
||||||
/// Request name
|
|
||||||
#[arg(short, long)]
|
|
||||||
name: Option<String>,
|
|
||||||
|
|
||||||
/// HTTP method
|
|
||||||
#[arg(short, long)]
|
|
||||||
method: Option<String>,
|
|
||||||
|
|
||||||
/// URL
|
|
||||||
#[arg(short, long)]
|
|
||||||
url: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload
|
|
||||||
#[arg(long)]
|
|
||||||
json: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Update an HTTP request
|
|
||||||
Update {
|
|
||||||
/// JSON payload
|
|
||||||
#[arg(long, conflicts_with = "json_input")]
|
|
||||||
json: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload shorthand
|
|
||||||
#[arg(value_name = "JSON", conflicts_with = "json")]
|
|
||||||
json_input: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Delete a request
|
|
||||||
Delete {
|
|
||||||
/// Request ID
|
|
||||||
request_id: String,
|
|
||||||
|
|
||||||
/// Skip confirmation prompt
|
|
||||||
#[arg(short, long)]
|
|
||||||
yes: bool,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, ValueEnum)]
|
|
||||||
pub enum RequestSchemaType {
|
|
||||||
Http,
|
|
||||||
Grpc,
|
|
||||||
Websocket,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, ValueEnum)]
|
|
||||||
pub enum LogLevel {
|
|
||||||
Error,
|
|
||||||
Warn,
|
|
||||||
Info,
|
|
||||||
Debug,
|
|
||||||
Trace,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl LogLevel {
|
|
||||||
pub fn as_filter(self) -> log::LevelFilter {
|
|
||||||
match self {
|
|
||||||
LogLevel::Error => log::LevelFilter::Error,
|
|
||||||
LogLevel::Warn => log::LevelFilter::Warn,
|
|
||||||
LogLevel::Info => log::LevelFilter::Info,
|
|
||||||
LogLevel::Debug => log::LevelFilter::Debug,
|
|
||||||
LogLevel::Trace => log::LevelFilter::Trace,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args)]
|
|
||||||
#[command(disable_help_subcommand = true)]
|
|
||||||
pub struct FolderArgs {
|
|
||||||
#[command(subcommand)]
|
|
||||||
pub command: FolderCommands,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand)]
|
|
||||||
pub enum FolderCommands {
|
|
||||||
/// List folders in a workspace
|
|
||||||
List {
|
|
||||||
/// Workspace ID
|
|
||||||
workspace_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Show a folder as JSON
|
|
||||||
Show {
|
|
||||||
/// Folder ID
|
|
||||||
folder_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Create a folder
|
|
||||||
Create {
|
|
||||||
/// Workspace ID (or positional JSON payload shorthand)
|
|
||||||
workspace_id: Option<String>,
|
|
||||||
|
|
||||||
/// Folder name
|
|
||||||
#[arg(short, long)]
|
|
||||||
name: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload
|
|
||||||
#[arg(long)]
|
|
||||||
json: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Update a folder
|
|
||||||
Update {
|
|
||||||
/// JSON payload
|
|
||||||
#[arg(long, conflicts_with = "json_input")]
|
|
||||||
json: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload shorthand
|
|
||||||
#[arg(value_name = "JSON", conflicts_with = "json")]
|
|
||||||
json_input: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Delete a folder
|
|
||||||
Delete {
|
|
||||||
/// Folder ID
|
|
||||||
folder_id: String,
|
|
||||||
|
|
||||||
/// Skip confirmation prompt
|
|
||||||
#[arg(short, long)]
|
|
||||||
yes: bool,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args)]
|
|
||||||
#[command(disable_help_subcommand = true)]
|
|
||||||
pub struct EnvironmentArgs {
|
|
||||||
#[command(subcommand)]
|
|
||||||
pub command: EnvironmentCommands,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand)]
|
|
||||||
pub enum EnvironmentCommands {
|
|
||||||
/// List environments in a workspace
|
|
||||||
List {
|
|
||||||
/// Workspace ID
|
|
||||||
workspace_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Output JSON schema for environment create/update payloads
|
|
||||||
Schema {
|
|
||||||
/// Pretty-print schema JSON output
|
|
||||||
#[arg(long)]
|
|
||||||
pretty: bool,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Show an environment as JSON
|
|
||||||
Show {
|
|
||||||
/// Environment ID
|
|
||||||
environment_id: String,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Create an environment
|
|
||||||
#[command(after_help = r#"Modes (choose one):
|
|
||||||
1) yaak environment create <workspace_id> --name <name>
|
|
||||||
2) yaak environment create --json '{"workspaceId":"wk_abc","name":"Production"}'
|
|
||||||
3) yaak environment create '{"workspaceId":"wk_abc","name":"Production"}'
|
|
||||||
4) yaak environment create <workspace_id> --json '{"name":"Production"}'
|
|
||||||
"#)]
|
|
||||||
Create {
|
|
||||||
/// Workspace ID for flag-based mode, or positional JSON payload shorthand
|
|
||||||
#[arg(value_name = "WORKSPACE_ID_OR_JSON")]
|
|
||||||
workspace_id: Option<String>,
|
|
||||||
|
|
||||||
/// Environment name
|
|
||||||
#[arg(short, long)]
|
|
||||||
name: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload (use instead of WORKSPACE_ID/--name)
|
|
||||||
#[arg(long)]
|
|
||||||
json: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Update an environment
|
|
||||||
Update {
|
|
||||||
/// JSON payload
|
|
||||||
#[arg(long, conflicts_with = "json_input")]
|
|
||||||
json: Option<String>,
|
|
||||||
|
|
||||||
/// JSON payload shorthand
|
|
||||||
#[arg(value_name = "JSON", conflicts_with = "json")]
|
|
||||||
json_input: Option<String>,
|
|
||||||
},
|
|
||||||
|
|
||||||
/// Delete an environment
|
|
||||||
Delete {
|
|
||||||
/// Environment ID
|
|
||||||
environment_id: String,
|
|
||||||
|
|
||||||
/// Skip confirmation prompt
|
|
||||||
#[arg(short, long)]
|
|
||||||
yes: bool,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args)]
|
|
||||||
#[command(disable_help_subcommand = true)]
|
|
||||||
pub struct AuthArgs {
|
|
||||||
#[command(subcommand)]
|
|
||||||
pub command: AuthCommands,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand)]
|
|
||||||
pub enum AuthCommands {
|
|
||||||
/// Login to Yaak via web browser
|
|
||||||
Login,
|
|
||||||
|
|
||||||
/// Sign out of the Yaak CLI
|
|
||||||
Logout,
|
|
||||||
|
|
||||||
/// Print the current logged-in user's info
|
|
||||||
Whoami,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args)]
|
|
||||||
#[command(disable_help_subcommand = true)]
|
|
||||||
pub struct PluginArgs {
|
|
||||||
#[command(subcommand)]
|
|
||||||
pub command: PluginCommands,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Subcommand)]
|
|
||||||
pub enum PluginCommands {
|
|
||||||
/// Transpile code into a runnable plugin bundle
|
|
||||||
Build(PluginPathArg),
|
|
||||||
|
|
||||||
/// Build plugin bundle continuously when the filesystem changes
|
|
||||||
Dev(PluginPathArg),
|
|
||||||
|
|
||||||
/// Generate a "Hello World" Yaak plugin
|
|
||||||
Generate(GenerateArgs),
|
|
||||||
|
|
||||||
/// Publish a Yaak plugin version to the plugin registry
|
|
||||||
Publish(PluginPathArg),
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args, Clone)]
|
|
||||||
pub struct PluginPathArg {
|
|
||||||
/// Path to plugin directory (defaults to current working directory)
|
|
||||||
pub path: Option<PathBuf>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Args, Clone)]
|
|
||||||
pub struct GenerateArgs {
|
|
||||||
/// Plugin name (defaults to a generated name in interactive mode)
|
|
||||||
#[arg(long)]
|
|
||||||
pub name: Option<String>,
|
|
||||||
|
|
||||||
/// Output directory for the generated plugin (defaults to ./<name> in interactive mode)
|
|
||||||
#[arg(long)]
|
|
||||||
pub dir: Option<PathBuf>,
|
|
||||||
}
|
|
||||||
@@ -1,528 +0,0 @@
|
|||||||
use crate::cli::{AuthArgs, AuthCommands};
|
|
||||||
use crate::ui;
|
|
||||||
use crate::utils::http;
|
|
||||||
use base64::Engine as _;
|
|
||||||
use keyring::Entry;
|
|
||||||
use rand::RngCore;
|
|
||||||
use rand::rngs::OsRng;
|
|
||||||
use reqwest::Url;
|
|
||||||
use serde_json::Value;
|
|
||||||
use sha2::{Digest, Sha256};
|
|
||||||
use std::io::{self, IsTerminal, Write};
|
|
||||||
use std::time::Duration;
|
|
||||||
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
|
||||||
use tokio::net::{TcpListener, TcpStream};
|
|
||||||
|
|
||||||
const OAUTH_CLIENT_ID: &str = "a1fe44800c2d7e803cad1b4bf07a291c";
|
|
||||||
const KEYRING_USER: &str = "yaak";
|
|
||||||
const AUTH_TIMEOUT: Duration = Duration::from_secs(300);
|
|
||||||
const MAX_REQUEST_BYTES: usize = 16 * 1024;
|
|
||||||
|
|
||||||
type CommandResult<T = ()> = std::result::Result<T, String>;
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
||||||
enum Environment {
|
|
||||||
Production,
|
|
||||||
Staging,
|
|
||||||
Development,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Environment {
|
|
||||||
fn app_base_url(self) -> &'static str {
|
|
||||||
match self {
|
|
||||||
Environment::Production => "https://yaak.app",
|
|
||||||
Environment::Staging => "https://todo.yaak.app",
|
|
||||||
Environment::Development => "http://localhost:9444",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn api_base_url(self) -> &'static str {
|
|
||||||
match self {
|
|
||||||
Environment::Production => "https://api.yaak.app",
|
|
||||||
Environment::Staging => "https://todo.yaak.app",
|
|
||||||
Environment::Development => "http://localhost:9444",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn keyring_service(self) -> &'static str {
|
|
||||||
match self {
|
|
||||||
Environment::Production => "app.yaak.cli.Token",
|
|
||||||
Environment::Staging => "app.yaak.cli.staging.Token",
|
|
||||||
Environment::Development => "app.yaak.cli.dev.Token",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct OAuthFlow {
|
|
||||||
app_base_url: String,
|
|
||||||
auth_url: Url,
|
|
||||||
token_url: String,
|
|
||||||
redirect_url: String,
|
|
||||||
state: String,
|
|
||||||
code_verifier: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn run(args: AuthArgs) -> i32 {
|
|
||||||
let result = match args.command {
|
|
||||||
AuthCommands::Login => login().await,
|
|
||||||
AuthCommands::Logout => logout(),
|
|
||||||
AuthCommands::Whoami => whoami().await,
|
|
||||||
};
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
ui::error(&error);
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn login() -> CommandResult {
|
|
||||||
let environment = current_environment();
|
|
||||||
|
|
||||||
let listener = TcpListener::bind("127.0.0.1:0")
|
|
||||||
.await
|
|
||||||
.map_err(|e| format!("Failed to start OAuth callback server: {e}"))?;
|
|
||||||
let port = listener
|
|
||||||
.local_addr()
|
|
||||||
.map_err(|e| format!("Failed to determine callback server port: {e}"))?
|
|
||||||
.port();
|
|
||||||
|
|
||||||
let oauth = build_oauth_flow(environment, port)?;
|
|
||||||
|
|
||||||
ui::info(&format!("Initiating login to {}", oauth.auth_url));
|
|
||||||
if !confirm_open_browser()? {
|
|
||||||
ui::info("Login canceled");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Err(err) = webbrowser::open(oauth.auth_url.as_ref()) {
|
|
||||||
ui::warning(&format!("Failed to open browser: {err}"));
|
|
||||||
ui::info(&format!("Open this URL manually:\n{}", oauth.auth_url));
|
|
||||||
}
|
|
||||||
ui::info("Waiting for authentication...");
|
|
||||||
|
|
||||||
let code = tokio::select! {
|
|
||||||
result = receive_oauth_code(listener, &oauth.state, &oauth.app_base_url) => result?,
|
|
||||||
_ = tokio::signal::ctrl_c() => {
|
|
||||||
return Err("Interrupted by user".to_string());
|
|
||||||
}
|
|
||||||
_ = tokio::time::sleep(AUTH_TIMEOUT) => {
|
|
||||||
return Err("Timeout waiting for authentication".to_string());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let token = exchange_access_token(&oauth, &code).await?;
|
|
||||||
store_auth_token(environment, &token)?;
|
|
||||||
ui::success("Authentication successful!");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn logout() -> CommandResult {
|
|
||||||
delete_auth_token(current_environment())?;
|
|
||||||
ui::success("Signed out of Yaak");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn whoami() -> CommandResult {
|
|
||||||
let environment = current_environment();
|
|
||||||
let token = match get_auth_token(environment)? {
|
|
||||||
Some(token) => token,
|
|
||||||
None => {
|
|
||||||
ui::warning("Not logged in");
|
|
||||||
ui::info("Please run `yaak auth login`");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let url = format!("{}/api/v1/whoami", environment.api_base_url());
|
|
||||||
let response = http::build_client(Some(&token))?
|
|
||||||
.get(url)
|
|
||||||
.send()
|
|
||||||
.await
|
|
||||||
.map_err(|e| format!("Failed to call whoami endpoint: {e}"))?;
|
|
||||||
|
|
||||||
let status = response.status();
|
|
||||||
let body =
|
|
||||||
response.text().await.map_err(|e| format!("Failed to read whoami response body: {e}"))?;
|
|
||||||
|
|
||||||
if !status.is_success() {
|
|
||||||
if status.as_u16() == 401 {
|
|
||||||
let _ = delete_auth_token(environment);
|
|
||||||
return Err(
|
|
||||||
"Unauthorized to access CLI. Run `yaak auth login` to refresh credentials."
|
|
||||||
.to_string(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return Err(http::parse_api_error(status.as_u16(), &body));
|
|
||||||
}
|
|
||||||
|
|
||||||
println!("{body}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn current_environment() -> Environment {
|
|
||||||
let value = std::env::var("ENVIRONMENT").ok();
|
|
||||||
parse_environment(value.as_deref())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_environment(value: Option<&str>) -> Environment {
|
|
||||||
match value {
|
|
||||||
Some("staging") => Environment::Staging,
|
|
||||||
Some("development") => Environment::Development,
|
|
||||||
_ => Environment::Production,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn build_oauth_flow(environment: Environment, callback_port: u16) -> CommandResult<OAuthFlow> {
|
|
||||||
let code_verifier = random_hex(32);
|
|
||||||
let state = random_hex(24);
|
|
||||||
let redirect_url = format!("http://127.0.0.1:{callback_port}/oauth/callback");
|
|
||||||
|
|
||||||
let code_challenge = base64::engine::general_purpose::URL_SAFE_NO_PAD
|
|
||||||
.encode(Sha256::digest(code_verifier.as_bytes()));
|
|
||||||
|
|
||||||
let mut auth_url = Url::parse(&format!("{}/login/oauth/authorize", environment.app_base_url()))
|
|
||||||
.map_err(|e| format!("Failed to build OAuth authorize URL: {e}"))?;
|
|
||||||
auth_url
|
|
||||||
.query_pairs_mut()
|
|
||||||
.append_pair("response_type", "code")
|
|
||||||
.append_pair("client_id", OAUTH_CLIENT_ID)
|
|
||||||
.append_pair("redirect_uri", &redirect_url)
|
|
||||||
.append_pair("state", &state)
|
|
||||||
.append_pair("code_challenge_method", "S256")
|
|
||||||
.append_pair("code_challenge", &code_challenge);
|
|
||||||
|
|
||||||
Ok(OAuthFlow {
|
|
||||||
app_base_url: environment.app_base_url().to_string(),
|
|
||||||
auth_url,
|
|
||||||
token_url: format!("{}/login/oauth/access_token", environment.app_base_url()),
|
|
||||||
redirect_url,
|
|
||||||
state,
|
|
||||||
code_verifier,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_oauth_code(
|
|
||||||
listener: TcpListener,
|
|
||||||
expected_state: &str,
|
|
||||||
app_base_url: &str,
|
|
||||||
) -> CommandResult<String> {
|
|
||||||
loop {
|
|
||||||
let (mut stream, _) = listener
|
|
||||||
.accept()
|
|
||||||
.await
|
|
||||||
.map_err(|e| format!("OAuth callback server accept error: {e}"))?;
|
|
||||||
|
|
||||||
match parse_callback_request(&mut stream).await {
|
|
||||||
Ok((state, code)) => {
|
|
||||||
if state != expected_state {
|
|
||||||
let _ = write_bad_request(&mut stream, "Invalid OAuth state").await;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let success_redirect = format!("{app_base_url}/login/oauth/success");
|
|
||||||
write_redirect(&mut stream, &success_redirect)
|
|
||||||
.await
|
|
||||||
.map_err(|e| format!("Failed responding to OAuth callback: {e}"))?;
|
|
||||||
return Ok(code);
|
|
||||||
}
|
|
||||||
Err(error) => {
|
|
||||||
let _ = write_bad_request(&mut stream, &error).await;
|
|
||||||
if error.starts_with("OAuth provider returned error:") {
|
|
||||||
return Err(error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn parse_callback_request(stream: &mut TcpStream) -> CommandResult<(String, String)> {
|
|
||||||
let target = read_http_target(stream).await?;
|
|
||||||
if !target.starts_with("/oauth/callback") {
|
|
||||||
return Err("Expected /oauth/callback path".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
let url = Url::parse(&format!("http://127.0.0.1{target}"))
|
|
||||||
.map_err(|e| format!("Failed to parse callback URL: {e}"))?;
|
|
||||||
let mut state: Option<String> = None;
|
|
||||||
let mut code: Option<String> = None;
|
|
||||||
let mut oauth_error: Option<String> = None;
|
|
||||||
let mut oauth_error_description: Option<String> = None;
|
|
||||||
|
|
||||||
for (k, v) in url.query_pairs() {
|
|
||||||
if k == "state" {
|
|
||||||
state = Some(v.into_owned());
|
|
||||||
} else if k == "code" {
|
|
||||||
code = Some(v.into_owned());
|
|
||||||
} else if k == "error" {
|
|
||||||
oauth_error = Some(v.into_owned());
|
|
||||||
} else if k == "error_description" {
|
|
||||||
oauth_error_description = Some(v.into_owned());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(error) = oauth_error {
|
|
||||||
let mut message = format!("OAuth provider returned error: {error}");
|
|
||||||
if let Some(description) = oauth_error_description.filter(|d| !d.is_empty()) {
|
|
||||||
message.push_str(&format!(" ({description})"));
|
|
||||||
}
|
|
||||||
return Err(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
let state = state.ok_or_else(|| "Missing 'state' query parameter".to_string())?;
|
|
||||||
let code = code.ok_or_else(|| "Missing 'code' query parameter".to_string())?;
|
|
||||||
|
|
||||||
if code.is_empty() {
|
|
||||||
return Err("Missing 'code' query parameter".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok((state, code))
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn read_http_target(stream: &mut TcpStream) -> CommandResult<String> {
|
|
||||||
let mut buf = vec![0_u8; MAX_REQUEST_BYTES];
|
|
||||||
let mut total_read = 0_usize;
|
|
||||||
|
|
||||||
loop {
|
|
||||||
let n = stream
|
|
||||||
.read(&mut buf[total_read..])
|
|
||||||
.await
|
|
||||||
.map_err(|e| format!("Failed reading callback request: {e}"))?;
|
|
||||||
if n == 0 {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
total_read += n;
|
|
||||||
|
|
||||||
if buf[..total_read].windows(4).any(|w| w == b"\r\n\r\n") {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if total_read == MAX_REQUEST_BYTES {
|
|
||||||
return Err("OAuth callback request too large".to_string());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let req = String::from_utf8_lossy(&buf[..total_read]);
|
|
||||||
let request_line =
|
|
||||||
req.lines().next().ok_or_else(|| "Invalid callback request line".to_string())?;
|
|
||||||
let mut parts = request_line.split_whitespace();
|
|
||||||
let method = parts.next().unwrap_or_default();
|
|
||||||
let target = parts.next().unwrap_or_default();
|
|
||||||
|
|
||||||
if method != "GET" {
|
|
||||||
return Err(format!("Expected GET callback request, got '{method}'"));
|
|
||||||
}
|
|
||||||
if target.is_empty() {
|
|
||||||
return Err("Missing callback request target".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(target.to_string())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn write_bad_request(stream: &mut TcpStream, message: &str) -> std::io::Result<()> {
|
|
||||||
let body = format!("Failed to authenticate: {message}");
|
|
||||||
let response = format!(
|
|
||||||
"HTTP/1.1 400 Bad Request\r\nContent-Type: text/plain; charset=utf-8\r\nContent-Length: {}\r\nConnection: close\r\n\r\n{}",
|
|
||||||
body.len(),
|
|
||||||
body
|
|
||||||
);
|
|
||||||
stream.write_all(response.as_bytes()).await?;
|
|
||||||
stream.shutdown().await
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn write_redirect(stream: &mut TcpStream, location: &str) -> std::io::Result<()> {
|
|
||||||
let response = format!(
|
|
||||||
"HTTP/1.1 302 Found\r\nLocation: {location}\r\nContent-Length: 0\r\nConnection: close\r\n\r\n"
|
|
||||||
);
|
|
||||||
stream.write_all(response.as_bytes()).await?;
|
|
||||||
stream.shutdown().await
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn exchange_access_token(oauth: &OAuthFlow, code: &str) -> CommandResult<String> {
|
|
||||||
let response = http::build_client(None)?
|
|
||||||
.post(&oauth.token_url)
|
|
||||||
.form(&[
|
|
||||||
("grant_type", "authorization_code"),
|
|
||||||
("client_id", OAUTH_CLIENT_ID),
|
|
||||||
("code", code),
|
|
||||||
("redirect_uri", oauth.redirect_url.as_str()),
|
|
||||||
("code_verifier", oauth.code_verifier.as_str()),
|
|
||||||
])
|
|
||||||
.send()
|
|
||||||
.await
|
|
||||||
.map_err(|e| format!("Failed to exchange OAuth code for access token: {e}"))?;
|
|
||||||
|
|
||||||
let status = response.status();
|
|
||||||
let body =
|
|
||||||
response.text().await.map_err(|e| format!("Failed to read token response body: {e}"))?;
|
|
||||||
|
|
||||||
if !status.is_success() {
|
|
||||||
return Err(format!(
|
|
||||||
"Failed to fetch access token: status={} body={}",
|
|
||||||
status.as_u16(),
|
|
||||||
body
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
let parsed: Value =
|
|
||||||
serde_json::from_str(&body).map_err(|e| format!("Invalid token response JSON: {e}"))?;
|
|
||||||
let token = parsed
|
|
||||||
.get("access_token")
|
|
||||||
.and_then(Value::as_str)
|
|
||||||
.filter(|s| !s.is_empty())
|
|
||||||
.ok_or_else(|| format!("Token response missing access_token: {body}"))?;
|
|
||||||
|
|
||||||
Ok(token.to_string())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn keyring_entry(environment: Environment) -> CommandResult<Entry> {
|
|
||||||
Entry::new(environment.keyring_service(), KEYRING_USER)
|
|
||||||
.map_err(|e| format!("Failed to initialize auth keyring entry: {e}"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_auth_token(environment: Environment) -> CommandResult<Option<String>> {
|
|
||||||
let entry = keyring_entry(environment)?;
|
|
||||||
match entry.get_password() {
|
|
||||||
Ok(token) => Ok(Some(token)),
|
|
||||||
Err(keyring::Error::NoEntry) => Ok(None),
|
|
||||||
Err(err) => Err(format!("Failed to read auth token: {err}")),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn store_auth_token(environment: Environment, token: &str) -> CommandResult {
|
|
||||||
let entry = keyring_entry(environment)?;
|
|
||||||
entry.set_password(token).map_err(|e| format!("Failed to store auth token: {e}"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn delete_auth_token(environment: Environment) -> CommandResult {
|
|
||||||
let entry = keyring_entry(environment)?;
|
|
||||||
match entry.delete_credential() {
|
|
||||||
Ok(()) | Err(keyring::Error::NoEntry) => Ok(()),
|
|
||||||
Err(err) => Err(format!("Failed to delete auth token: {err}")),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn random_hex(bytes: usize) -> String {
|
|
||||||
let mut data = vec![0_u8; bytes];
|
|
||||||
OsRng.fill_bytes(&mut data);
|
|
||||||
hex::encode(data)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn confirm_open_browser() -> CommandResult<bool> {
|
|
||||||
if !io::stdin().is_terminal() {
|
|
||||||
return Ok(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
loop {
|
|
||||||
print!("Open default browser? [Y/n]: ");
|
|
||||||
io::stdout().flush().map_err(|e| format!("Failed to flush stdout: {e}"))?;
|
|
||||||
|
|
||||||
let mut input = String::new();
|
|
||||||
io::stdin().read_line(&mut input).map_err(|e| format!("Failed to read input: {e}"))?;
|
|
||||||
|
|
||||||
match input.trim().to_ascii_lowercase().as_str() {
|
|
||||||
"" | "y" | "yes" => return Ok(true),
|
|
||||||
"n" | "no" => return Ok(false),
|
|
||||||
_ => ui::warning("Please answer y or n"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod tests {
|
|
||||||
use super::*;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn environment_mapping() {
|
|
||||||
assert_eq!(parse_environment(Some("staging")), Environment::Staging);
|
|
||||||
assert_eq!(parse_environment(Some("development")), Environment::Development);
|
|
||||||
assert_eq!(parse_environment(Some("production")), Environment::Production);
|
|
||||||
assert_eq!(parse_environment(None), Environment::Production);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
async fn parses_callback_request() {
|
|
||||||
let listener = TcpListener::bind("127.0.0.1:0").await.expect("bind");
|
|
||||||
let addr = listener.local_addr().expect("local addr");
|
|
||||||
|
|
||||||
let server = tokio::spawn(async move {
|
|
||||||
let (mut stream, _) = listener.accept().await.expect("accept");
|
|
||||||
parse_callback_request(&mut stream).await
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut client = TcpStream::connect(addr).await.expect("connect");
|
|
||||||
client
|
|
||||||
.write_all(
|
|
||||||
b"GET /oauth/callback?code=abc123&state=xyz HTTP/1.1\r\nHost: localhost\r\n\r\n",
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.expect("write");
|
|
||||||
|
|
||||||
let parsed = server.await.expect("join").expect("parse");
|
|
||||||
assert_eq!(parsed.0, "xyz");
|
|
||||||
assert_eq!(parsed.1, "abc123");
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
async fn parse_callback_request_oauth_error() {
|
|
||||||
let listener = TcpListener::bind("127.0.0.1:0").await.expect("bind");
|
|
||||||
let addr = listener.local_addr().expect("local addr");
|
|
||||||
|
|
||||||
let server = tokio::spawn(async move {
|
|
||||||
let (mut stream, _) = listener.accept().await.expect("accept");
|
|
||||||
parse_callback_request(&mut stream).await
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut client = TcpStream::connect(addr).await.expect("connect");
|
|
||||||
client
|
|
||||||
.write_all(
|
|
||||||
b"GET /oauth/callback?error=access_denied&error_description=User%20denied&state=xyz HTTP/1.1\r\nHost: localhost\r\n\r\n",
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.expect("write");
|
|
||||||
|
|
||||||
let err = server.await.expect("join").expect_err("should fail");
|
|
||||||
assert!(err.contains("OAuth provider returned error: access_denied"));
|
|
||||||
assert!(err.contains("User denied"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tokio::test]
|
|
||||||
async fn receive_oauth_code_fails_fast_on_provider_error() {
|
|
||||||
let listener = TcpListener::bind("127.0.0.1:0").await.expect("bind");
|
|
||||||
let addr = listener.local_addr().expect("local addr");
|
|
||||||
|
|
||||||
let server = tokio::spawn(async move {
|
|
||||||
receive_oauth_code(listener, "expected-state", "http://localhost:9444").await
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut client = TcpStream::connect(addr).await.expect("connect");
|
|
||||||
client
|
|
||||||
.write_all(
|
|
||||||
b"GET /oauth/callback?error=access_denied&state=expected-state HTTP/1.1\r\nHost: localhost\r\n\r\n",
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.expect("write");
|
|
||||||
|
|
||||||
let result = tokio::time::timeout(std::time::Duration::from_secs(2), server)
|
|
||||||
.await
|
|
||||||
.expect("should not timeout")
|
|
||||||
.expect("join");
|
|
||||||
let err = result.expect_err("should return oauth error");
|
|
||||||
assert!(err.contains("OAuth provider returned error: access_denied"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn builds_oauth_flow_with_pkce() {
|
|
||||||
let flow = build_oauth_flow(Environment::Development, 8080).expect("flow");
|
|
||||||
assert!(flow.auth_url.as_str().contains("code_challenge_method=S256"));
|
|
||||||
assert!(
|
|
||||||
flow.auth_url
|
|
||||||
.as_str()
|
|
||||||
.contains("redirect_uri=http%3A%2F%2F127.0.0.1%3A8080%2Foauth%2Fcallback")
|
|
||||||
);
|
|
||||||
assert_eq!(flow.redirect_url, "http://127.0.0.1:8080/oauth/callback");
|
|
||||||
assert_eq!(flow.token_url, "http://localhost:9444/login/oauth/access_token");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,169 +0,0 @@
|
|||||||
use crate::cli::{EnvironmentArgs, EnvironmentCommands};
|
|
||||||
use crate::context::CliContext;
|
|
||||||
use crate::utils::confirm::confirm_delete;
|
|
||||||
use crate::utils::json::{
|
|
||||||
apply_merge_patch, is_json_shorthand, merge_workspace_id_arg, parse_optional_json,
|
|
||||||
parse_required_json, require_id, validate_create_id,
|
|
||||||
};
|
|
||||||
use crate::utils::schema::append_agent_hints;
|
|
||||||
use schemars::schema_for;
|
|
||||||
use yaak_models::models::Environment;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
type CommandResult<T = ()> = std::result::Result<T, String>;
|
|
||||||
|
|
||||||
pub fn run(ctx: &CliContext, args: EnvironmentArgs) -> i32 {
|
|
||||||
let result = match args.command {
|
|
||||||
EnvironmentCommands::List { workspace_id } => list(ctx, &workspace_id),
|
|
||||||
EnvironmentCommands::Schema { pretty } => schema(pretty),
|
|
||||||
EnvironmentCommands::Show { environment_id } => show(ctx, &environment_id),
|
|
||||||
EnvironmentCommands::Create { workspace_id, name, json } => {
|
|
||||||
create(ctx, workspace_id, name, json)
|
|
||||||
}
|
|
||||||
EnvironmentCommands::Update { json, json_input } => update(ctx, json, json_input),
|
|
||||||
EnvironmentCommands::Delete { environment_id, yes } => delete(ctx, &environment_id, yes),
|
|
||||||
};
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!("Error: {error}");
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn schema(pretty: bool) -> CommandResult {
|
|
||||||
let mut schema = serde_json::to_value(schema_for!(Environment))
|
|
||||||
.map_err(|e| format!("Failed to serialize environment schema: {e}"))?;
|
|
||||||
append_agent_hints(&mut schema);
|
|
||||||
|
|
||||||
let output =
|
|
||||||
if pretty { serde_json::to_string_pretty(&schema) } else { serde_json::to_string(&schema) }
|
|
||||||
.map_err(|e| format!("Failed to format environment schema JSON: {e}"))?;
|
|
||||||
println!("{output}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn list(ctx: &CliContext, workspace_id: &str) -> CommandResult {
|
|
||||||
let environments = ctx
|
|
||||||
.db()
|
|
||||||
.list_environments_ensure_base(workspace_id)
|
|
||||||
.map_err(|e| format!("Failed to list environments: {e}"))?;
|
|
||||||
|
|
||||||
if environments.is_empty() {
|
|
||||||
println!("No environments found in workspace {}", workspace_id);
|
|
||||||
} else {
|
|
||||||
for environment in environments {
|
|
||||||
println!("{} - {} ({})", environment.id, environment.name, environment.parent_model);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn show(ctx: &CliContext, environment_id: &str) -> CommandResult {
|
|
||||||
let environment = ctx
|
|
||||||
.db()
|
|
||||||
.get_environment(environment_id)
|
|
||||||
.map_err(|e| format!("Failed to get environment: {e}"))?;
|
|
||||||
let output = serde_json::to_string_pretty(&environment)
|
|
||||||
.map_err(|e| format!("Failed to serialize environment: {e}"))?;
|
|
||||||
println!("{output}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn create(
|
|
||||||
ctx: &CliContext,
|
|
||||||
workspace_id: Option<String>,
|
|
||||||
name: Option<String>,
|
|
||||||
json: Option<String>,
|
|
||||||
) -> CommandResult {
|
|
||||||
let json_shorthand =
|
|
||||||
workspace_id.as_deref().filter(|v| is_json_shorthand(v)).map(str::to_owned);
|
|
||||||
let workspace_id_arg = workspace_id.filter(|v| !is_json_shorthand(v));
|
|
||||||
|
|
||||||
let payload = parse_optional_json(json, json_shorthand, "environment create")?;
|
|
||||||
|
|
||||||
if let Some(payload) = payload {
|
|
||||||
if name.is_some() {
|
|
||||||
return Err("environment create cannot combine --name with JSON payload".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
validate_create_id(&payload, "environment")?;
|
|
||||||
let mut environment: Environment = serde_json::from_value(payload)
|
|
||||||
.map_err(|e| format!("Failed to parse environment create JSON: {e}"))?;
|
|
||||||
merge_workspace_id_arg(
|
|
||||||
workspace_id_arg.as_deref(),
|
|
||||||
&mut environment.workspace_id,
|
|
||||||
"environment create",
|
|
||||||
)?;
|
|
||||||
|
|
||||||
if environment.parent_model.is_empty() {
|
|
||||||
environment.parent_model = "environment".to_string();
|
|
||||||
}
|
|
||||||
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_environment(&environment, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create environment: {e}"))?;
|
|
||||||
|
|
||||||
println!("Created environment: {}", created.id);
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let workspace_id = workspace_id_arg.ok_or_else(|| {
|
|
||||||
"environment create requires workspace_id unless JSON payload is provided".to_string()
|
|
||||||
})?;
|
|
||||||
let name = name.ok_or_else(|| {
|
|
||||||
"environment create requires --name unless JSON payload is provided".to_string()
|
|
||||||
})?;
|
|
||||||
|
|
||||||
let environment = Environment {
|
|
||||||
workspace_id,
|
|
||||||
name,
|
|
||||||
parent_model: "environment".to_string(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_environment(&environment, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create environment: {e}"))?;
|
|
||||||
|
|
||||||
println!("Created environment: {}", created.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update(ctx: &CliContext, json: Option<String>, json_input: Option<String>) -> CommandResult {
|
|
||||||
let patch = parse_required_json(json, json_input, "environment update")?;
|
|
||||||
let id = require_id(&patch, "environment update")?;
|
|
||||||
|
|
||||||
let existing = ctx
|
|
||||||
.db()
|
|
||||||
.get_environment(&id)
|
|
||||||
.map_err(|e| format!("Failed to get environment for update: {e}"))?;
|
|
||||||
let updated = apply_merge_patch(&existing, &patch, &id, "environment update")?;
|
|
||||||
|
|
||||||
let saved = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_environment(&updated, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to update environment: {e}"))?;
|
|
||||||
|
|
||||||
println!("Updated environment: {}", saved.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn delete(ctx: &CliContext, environment_id: &str, yes: bool) -> CommandResult {
|
|
||||||
if !yes && !confirm_delete("environment", environment_id) {
|
|
||||||
println!("Aborted");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let deleted = ctx
|
|
||||||
.db()
|
|
||||||
.delete_environment_by_id(environment_id, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to delete environment: {e}"))?;
|
|
||||||
|
|
||||||
println!("Deleted environment: {}", deleted.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
@@ -1,138 +0,0 @@
|
|||||||
use crate::cli::{FolderArgs, FolderCommands};
|
|
||||||
use crate::context::CliContext;
|
|
||||||
use crate::utils::confirm::confirm_delete;
|
|
||||||
use crate::utils::json::{
|
|
||||||
apply_merge_patch, is_json_shorthand, merge_workspace_id_arg, parse_optional_json,
|
|
||||||
parse_required_json, require_id, validate_create_id,
|
|
||||||
};
|
|
||||||
use yaak_models::models::Folder;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
type CommandResult<T = ()> = std::result::Result<T, String>;
|
|
||||||
|
|
||||||
pub fn run(ctx: &CliContext, args: FolderArgs) -> i32 {
|
|
||||||
let result = match args.command {
|
|
||||||
FolderCommands::List { workspace_id } => list(ctx, &workspace_id),
|
|
||||||
FolderCommands::Show { folder_id } => show(ctx, &folder_id),
|
|
||||||
FolderCommands::Create { workspace_id, name, json } => {
|
|
||||||
create(ctx, workspace_id, name, json)
|
|
||||||
}
|
|
||||||
FolderCommands::Update { json, json_input } => update(ctx, json, json_input),
|
|
||||||
FolderCommands::Delete { folder_id, yes } => delete(ctx, &folder_id, yes),
|
|
||||||
};
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!("Error: {error}");
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn list(ctx: &CliContext, workspace_id: &str) -> CommandResult {
|
|
||||||
let folders =
|
|
||||||
ctx.db().list_folders(workspace_id).map_err(|e| format!("Failed to list folders: {e}"))?;
|
|
||||||
if folders.is_empty() {
|
|
||||||
println!("No folders found in workspace {}", workspace_id);
|
|
||||||
} else {
|
|
||||||
for folder in folders {
|
|
||||||
println!("{} - {}", folder.id, folder.name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn show(ctx: &CliContext, folder_id: &str) -> CommandResult {
|
|
||||||
let folder =
|
|
||||||
ctx.db().get_folder(folder_id).map_err(|e| format!("Failed to get folder: {e}"))?;
|
|
||||||
let output = serde_json::to_string_pretty(&folder)
|
|
||||||
.map_err(|e| format!("Failed to serialize folder: {e}"))?;
|
|
||||||
println!("{output}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn create(
|
|
||||||
ctx: &CliContext,
|
|
||||||
workspace_id: Option<String>,
|
|
||||||
name: Option<String>,
|
|
||||||
json: Option<String>,
|
|
||||||
) -> CommandResult {
|
|
||||||
let json_shorthand =
|
|
||||||
workspace_id.as_deref().filter(|v| is_json_shorthand(v)).map(str::to_owned);
|
|
||||||
let workspace_id_arg = workspace_id.filter(|v| !is_json_shorthand(v));
|
|
||||||
|
|
||||||
let payload = parse_optional_json(json, json_shorthand, "folder create")?;
|
|
||||||
|
|
||||||
if let Some(payload) = payload {
|
|
||||||
if name.is_some() {
|
|
||||||
return Err("folder create cannot combine --name with JSON payload".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
validate_create_id(&payload, "folder")?;
|
|
||||||
let mut folder: Folder = serde_json::from_value(payload)
|
|
||||||
.map_err(|e| format!("Failed to parse folder create JSON: {e}"))?;
|
|
||||||
merge_workspace_id_arg(
|
|
||||||
workspace_id_arg.as_deref(),
|
|
||||||
&mut folder.workspace_id,
|
|
||||||
"folder create",
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_folder(&folder, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create folder: {e}"))?;
|
|
||||||
|
|
||||||
println!("Created folder: {}", created.id);
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let workspace_id = workspace_id_arg.ok_or_else(|| {
|
|
||||||
"folder create requires workspace_id unless JSON payload is provided".to_string()
|
|
||||||
})?;
|
|
||||||
let name = name.ok_or_else(|| {
|
|
||||||
"folder create requires --name unless JSON payload is provided".to_string()
|
|
||||||
})?;
|
|
||||||
|
|
||||||
let folder = Folder { workspace_id, name, ..Default::default() };
|
|
||||||
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_folder(&folder, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create folder: {e}"))?;
|
|
||||||
|
|
||||||
println!("Created folder: {}", created.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update(ctx: &CliContext, json: Option<String>, json_input: Option<String>) -> CommandResult {
|
|
||||||
let patch = parse_required_json(json, json_input, "folder update")?;
|
|
||||||
let id = require_id(&patch, "folder update")?;
|
|
||||||
|
|
||||||
let existing =
|
|
||||||
ctx.db().get_folder(&id).map_err(|e| format!("Failed to get folder for update: {e}"))?;
|
|
||||||
let updated = apply_merge_patch(&existing, &patch, &id, "folder update")?;
|
|
||||||
|
|
||||||
let saved = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_folder(&updated, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to update folder: {e}"))?;
|
|
||||||
|
|
||||||
println!("Updated folder: {}", saved.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn delete(ctx: &CliContext, folder_id: &str, yes: bool) -> CommandResult {
|
|
||||||
if !yes && !confirm_delete("folder", folder_id) {
|
|
||||||
println!("Aborted");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let deleted = ctx
|
|
||||||
.db()
|
|
||||||
.delete_folder_by_id(folder_id, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to delete folder: {e}"))?;
|
|
||||||
|
|
||||||
println!("Deleted folder: {}", deleted.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
pub mod auth;
|
|
||||||
pub mod environment;
|
|
||||||
pub mod folder;
|
|
||||||
pub mod plugin;
|
|
||||||
pub mod request;
|
|
||||||
pub mod send;
|
|
||||||
pub mod workspace;
|
|
||||||
@@ -1,525 +0,0 @@
|
|||||||
use crate::cli::{GenerateArgs, PluginArgs, PluginCommands, PluginPathArg};
|
|
||||||
use crate::ui;
|
|
||||||
use crate::utils::http;
|
|
||||||
use keyring::Entry;
|
|
||||||
use rand::Rng;
|
|
||||||
use rolldown::{
|
|
||||||
Bundler, BundlerOptions, ExperimentalOptions, InputItem, LogLevel, OutputFormat, Platform,
|
|
||||||
WatchOption, Watcher,
|
|
||||||
};
|
|
||||||
use serde::Deserialize;
|
|
||||||
use std::collections::HashSet;
|
|
||||||
use std::fs;
|
|
||||||
use std::io::{self, IsTerminal, Read, Write};
|
|
||||||
use std::path::{Path, PathBuf};
|
|
||||||
use std::sync::Arc;
|
|
||||||
use tokio::sync::Mutex;
|
|
||||||
use walkdir::WalkDir;
|
|
||||||
use zip::CompressionMethod;
|
|
||||||
use zip::write::SimpleFileOptions;
|
|
||||||
|
|
||||||
type CommandResult<T = ()> = std::result::Result<T, String>;
|
|
||||||
|
|
||||||
const KEYRING_USER: &str = "yaak";
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
||||||
enum Environment {
|
|
||||||
Production,
|
|
||||||
Staging,
|
|
||||||
Development,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Environment {
|
|
||||||
fn api_base_url(self) -> &'static str {
|
|
||||||
match self {
|
|
||||||
Environment::Production => "https://api.yaak.app",
|
|
||||||
Environment::Staging => "https://todo.yaak.app",
|
|
||||||
Environment::Development => "http://localhost:9444",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn keyring_service(self) -> &'static str {
|
|
||||||
match self {
|
|
||||||
Environment::Production => "app.yaak.cli.Token",
|
|
||||||
Environment::Staging => "app.yaak.cli.staging.Token",
|
|
||||||
Environment::Development => "app.yaak.cli.dev.Token",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn run_build(args: PluginPathArg) -> i32 {
|
|
||||||
match build(args).await {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
ui::error(&error);
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn run(args: PluginArgs) -> i32 {
|
|
||||||
match args.command {
|
|
||||||
PluginCommands::Build(args) => run_build(args).await,
|
|
||||||
PluginCommands::Dev(args) => run_dev(args).await,
|
|
||||||
PluginCommands::Generate(args) => run_generate(args).await,
|
|
||||||
PluginCommands::Publish(args) => run_publish(args).await,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn run_dev(args: PluginPathArg) -> i32 {
|
|
||||||
match dev(args).await {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
ui::error(&error);
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn run_generate(args: GenerateArgs) -> i32 {
|
|
||||||
match generate(args) {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
ui::error(&error);
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn run_publish(args: PluginPathArg) -> i32 {
|
|
||||||
match publish(args).await {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
ui::error(&error);
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn build(args: PluginPathArg) -> CommandResult {
|
|
||||||
let plugin_dir = resolve_plugin_dir(args.path)?;
|
|
||||||
ensure_plugin_build_inputs(&plugin_dir)?;
|
|
||||||
|
|
||||||
ui::info(&format!("Building plugin {}...", plugin_dir.display()));
|
|
||||||
let warnings = build_plugin_bundle(&plugin_dir).await?;
|
|
||||||
for warning in warnings {
|
|
||||||
ui::warning(&warning);
|
|
||||||
}
|
|
||||||
ui::success(&format!("Built plugin bundle at {}", plugin_dir.join("build/index.js").display()));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn dev(args: PluginPathArg) -> CommandResult {
|
|
||||||
let plugin_dir = resolve_plugin_dir(args.path)?;
|
|
||||||
ensure_plugin_build_inputs(&plugin_dir)?;
|
|
||||||
|
|
||||||
ui::info(&format!("Watching plugin {}...", plugin_dir.display()));
|
|
||||||
ui::info("Press Ctrl-C to stop");
|
|
||||||
|
|
||||||
let bundler = Bundler::new(bundler_options(&plugin_dir, true))
|
|
||||||
.map_err(|err| format!("Failed to initialize Rolldown watcher: {err}"))?;
|
|
||||||
let watcher = Watcher::new(vec![Arc::new(Mutex::new(bundler))], None)
|
|
||||||
.map_err(|err| format!("Failed to start Rolldown watcher: {err}"))?;
|
|
||||||
|
|
||||||
watcher.start().await;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn generate(args: GenerateArgs) -> CommandResult {
|
|
||||||
let default_name = random_name();
|
|
||||||
let name = match args.name {
|
|
||||||
Some(name) => name,
|
|
||||||
None => prompt_with_default("Plugin name", &default_name)?,
|
|
||||||
};
|
|
||||||
|
|
||||||
let default_dir = format!("./{name}");
|
|
||||||
let output_dir = match args.dir {
|
|
||||||
Some(dir) => dir,
|
|
||||||
None => PathBuf::from(prompt_with_default("Plugin dir", &default_dir)?),
|
|
||||||
};
|
|
||||||
|
|
||||||
if output_dir.exists() {
|
|
||||||
return Err(format!("Plugin directory already exists: {}", output_dir.display()));
|
|
||||||
}
|
|
||||||
|
|
||||||
ui::info(&format!("Generating plugin in {}", output_dir.display()));
|
|
||||||
fs::create_dir_all(output_dir.join("src"))
|
|
||||||
.map_err(|e| format!("Failed creating plugin directory {}: {e}", output_dir.display()))?;
|
|
||||||
|
|
||||||
write_file(&output_dir.join(".gitignore"), TEMPLATE_GITIGNORE)?;
|
|
||||||
write_file(
|
|
||||||
&output_dir.join("package.json"),
|
|
||||||
&TEMPLATE_PACKAGE_JSON.replace("yaak-plugin-name", &name),
|
|
||||||
)?;
|
|
||||||
write_file(&output_dir.join("tsconfig.json"), TEMPLATE_TSCONFIG)?;
|
|
||||||
write_file(&output_dir.join("README.md"), &TEMPLATE_README.replace("yaak-plugin-name", &name))?;
|
|
||||||
write_file(
|
|
||||||
&output_dir.join("src/index.ts"),
|
|
||||||
&TEMPLATE_INDEX_TS.replace("yaak-plugin-name", &name),
|
|
||||||
)?;
|
|
||||||
write_file(&output_dir.join("src/index.test.ts"), TEMPLATE_INDEX_TEST_TS)?;
|
|
||||||
|
|
||||||
ui::success("Plugin scaffold generated");
|
|
||||||
ui::info("Next steps:");
|
|
||||||
println!(" 1. cd {}", output_dir.display());
|
|
||||||
println!(" 2. npm install");
|
|
||||||
println!(" 3. yaak plugin build");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn publish(args: PluginPathArg) -> CommandResult {
|
|
||||||
let plugin_dir = resolve_plugin_dir(args.path)?;
|
|
||||||
ensure_plugin_build_inputs(&plugin_dir)?;
|
|
||||||
|
|
||||||
let environment = current_environment();
|
|
||||||
let token = get_auth_token(environment)?
|
|
||||||
.ok_or_else(|| "Not logged in. Run `yaak auth login`.".to_string())?;
|
|
||||||
|
|
||||||
ui::info(&format!("Building plugin {}...", plugin_dir.display()));
|
|
||||||
let warnings = build_plugin_bundle(&plugin_dir).await?;
|
|
||||||
for warning in warnings {
|
|
||||||
ui::warning(&warning);
|
|
||||||
}
|
|
||||||
|
|
||||||
ui::info("Archiving plugin");
|
|
||||||
let archive = create_publish_archive(&plugin_dir)?;
|
|
||||||
|
|
||||||
ui::info("Uploading plugin");
|
|
||||||
let url = format!("{}/api/v1/plugins/publish", environment.api_base_url());
|
|
||||||
let response = http::build_client(Some(&token))?
|
|
||||||
.post(url)
|
|
||||||
.header(reqwest::header::CONTENT_TYPE, "application/zip")
|
|
||||||
.body(archive)
|
|
||||||
.send()
|
|
||||||
.await
|
|
||||||
.map_err(|e| format!("Failed to upload plugin: {e}"))?;
|
|
||||||
|
|
||||||
let status = response.status();
|
|
||||||
let body =
|
|
||||||
response.text().await.map_err(|e| format!("Failed reading publish response body: {e}"))?;
|
|
||||||
|
|
||||||
if !status.is_success() {
|
|
||||||
return Err(http::parse_api_error(status.as_u16(), &body));
|
|
||||||
}
|
|
||||||
|
|
||||||
let published: PublishResponse = serde_json::from_str(&body)
|
|
||||||
.map_err(|e| format!("Failed parsing publish response JSON: {e}\nResponse: {body}"))?;
|
|
||||||
ui::success(&format!("Plugin published {}", published.version));
|
|
||||||
println!(" -> {}", published.url);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize)]
|
|
||||||
struct PublishResponse {
|
|
||||||
version: String,
|
|
||||||
url: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn build_plugin_bundle(plugin_dir: &Path) -> CommandResult<Vec<String>> {
|
|
||||||
prepare_build_output_dir(plugin_dir)?;
|
|
||||||
let mut bundler = Bundler::new(bundler_options(plugin_dir, false))
|
|
||||||
.map_err(|err| format!("Failed to initialize Rolldown: {err}"))?;
|
|
||||||
let output = bundler.write().await.map_err(|err| format!("Plugin build failed:\n{err}"))?;
|
|
||||||
|
|
||||||
Ok(output.warnings.into_iter().map(|w| w.to_string()).collect())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn prepare_build_output_dir(plugin_dir: &Path) -> CommandResult {
|
|
||||||
let build_dir = plugin_dir.join("build");
|
|
||||||
if build_dir.exists() {
|
|
||||||
fs::remove_dir_all(&build_dir)
|
|
||||||
.map_err(|e| format!("Failed to clean build directory {}: {e}", build_dir.display()))?;
|
|
||||||
}
|
|
||||||
fs::create_dir_all(&build_dir)
|
|
||||||
.map_err(|e| format!("Failed to create build directory {}: {e}", build_dir.display()))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn bundler_options(plugin_dir: &Path, watch: bool) -> BundlerOptions {
|
|
||||||
BundlerOptions {
|
|
||||||
input: Some(vec![InputItem { import: "./src/index.ts".to_string(), ..Default::default() }]),
|
|
||||||
cwd: Some(plugin_dir.to_path_buf()),
|
|
||||||
file: Some("build/index.js".to_string()),
|
|
||||||
format: Some(OutputFormat::Cjs),
|
|
||||||
platform: Some(Platform::Node),
|
|
||||||
log_level: Some(LogLevel::Info),
|
|
||||||
experimental: watch
|
|
||||||
.then_some(ExperimentalOptions { incremental_build: Some(true), ..Default::default() }),
|
|
||||||
watch: watch.then_some(WatchOption::default()),
|
|
||||||
..Default::default()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn resolve_plugin_dir(path: Option<PathBuf>) -> CommandResult<PathBuf> {
|
|
||||||
let cwd =
|
|
||||||
std::env::current_dir().map_err(|e| format!("Failed to read current directory: {e}"))?;
|
|
||||||
let candidate = match path {
|
|
||||||
Some(path) if path.is_absolute() => path,
|
|
||||||
Some(path) => cwd.join(path),
|
|
||||||
None => cwd,
|
|
||||||
};
|
|
||||||
|
|
||||||
if !candidate.exists() {
|
|
||||||
return Err(format!("Plugin directory does not exist: {}", candidate.display()));
|
|
||||||
}
|
|
||||||
if !candidate.is_dir() {
|
|
||||||
return Err(format!("Plugin path is not a directory: {}", candidate.display()));
|
|
||||||
}
|
|
||||||
|
|
||||||
candidate
|
|
||||||
.canonicalize()
|
|
||||||
.map_err(|e| format!("Failed to resolve plugin directory {}: {e}", candidate.display()))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn ensure_plugin_build_inputs(plugin_dir: &Path) -> CommandResult {
|
|
||||||
let package_json = plugin_dir.join("package.json");
|
|
||||||
if !package_json.is_file() {
|
|
||||||
return Err(format!(
|
|
||||||
"{} does not exist. Ensure that you are in a plugin directory.",
|
|
||||||
package_json.display()
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
let entry = plugin_dir.join("src/index.ts");
|
|
||||||
if !entry.is_file() {
|
|
||||||
return Err(format!("Required entrypoint missing: {}", entry.display()));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn create_publish_archive(plugin_dir: &Path) -> CommandResult<Vec<u8>> {
|
|
||||||
let required_files = [
|
|
||||||
"README.md",
|
|
||||||
"package.json",
|
|
||||||
"build/index.js",
|
|
||||||
"src/index.ts",
|
|
||||||
];
|
|
||||||
let optional_files = ["package-lock.json"];
|
|
||||||
|
|
||||||
let mut selected = HashSet::new();
|
|
||||||
for required in required_files {
|
|
||||||
let required_path = plugin_dir.join(required);
|
|
||||||
if !required_path.is_file() {
|
|
||||||
return Err(format!("Missing required file: {required}"));
|
|
||||||
}
|
|
||||||
selected.insert(required.to_string());
|
|
||||||
}
|
|
||||||
for optional in optional_files {
|
|
||||||
selected.insert(optional.to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
let cursor = std::io::Cursor::new(Vec::new());
|
|
||||||
let mut zip = zip::ZipWriter::new(cursor);
|
|
||||||
let options = SimpleFileOptions::default().compression_method(CompressionMethod::Deflated);
|
|
||||||
|
|
||||||
for entry in WalkDir::new(plugin_dir) {
|
|
||||||
let entry = entry.map_err(|e| format!("Failed walking plugin directory: {e}"))?;
|
|
||||||
if !entry.file_type().is_file() {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let path = entry.path();
|
|
||||||
let rel = path
|
|
||||||
.strip_prefix(plugin_dir)
|
|
||||||
.map_err(|e| format!("Failed deriving relative path for {}: {e}", path.display()))?;
|
|
||||||
let rel = rel.to_string_lossy().replace('\\', "/");
|
|
||||||
|
|
||||||
let keep = rel.starts_with("src/") || rel.starts_with("build/") || selected.contains(&rel);
|
|
||||||
if !keep {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
zip.start_file(rel, options).map_err(|e| format!("Failed adding file to archive: {e}"))?;
|
|
||||||
let mut file = fs::File::open(path)
|
|
||||||
.map_err(|e| format!("Failed opening file {}: {e}", path.display()))?;
|
|
||||||
let mut contents = Vec::new();
|
|
||||||
file.read_to_end(&mut contents)
|
|
||||||
.map_err(|e| format!("Failed reading file {}: {e}", path.display()))?;
|
|
||||||
zip.write_all(&contents).map_err(|e| format!("Failed writing archive contents: {e}"))?;
|
|
||||||
}
|
|
||||||
|
|
||||||
let cursor = zip.finish().map_err(|e| format!("Failed finalizing plugin archive: {e}"))?;
|
|
||||||
Ok(cursor.into_inner())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn write_file(path: &Path, contents: &str) -> CommandResult {
|
|
||||||
if let Some(parent) = path.parent() {
|
|
||||||
fs::create_dir_all(parent)
|
|
||||||
.map_err(|e| format!("Failed creating directory {}: {e}", parent.display()))?;
|
|
||||||
}
|
|
||||||
fs::write(path, contents).map_err(|e| format!("Failed writing file {}: {e}", path.display()))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn prompt_with_default(label: &str, default: &str) -> CommandResult<String> {
|
|
||||||
if !io::stdin().is_terminal() {
|
|
||||||
return Ok(default.to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
print!("{label} [{default}]: ");
|
|
||||||
io::stdout().flush().map_err(|e| format!("Failed to flush stdout: {e}"))?;
|
|
||||||
|
|
||||||
let mut input = String::new();
|
|
||||||
io::stdin().read_line(&mut input).map_err(|e| format!("Failed to read input: {e}"))?;
|
|
||||||
let trimmed = input.trim();
|
|
||||||
|
|
||||||
if trimmed.is_empty() { Ok(default.to_string()) } else { Ok(trimmed.to_string()) }
|
|
||||||
}
|
|
||||||
|
|
||||||
fn current_environment() -> Environment {
|
|
||||||
match std::env::var("ENVIRONMENT").as_deref() {
|
|
||||||
Ok("staging") => Environment::Staging,
|
|
||||||
Ok("development") => Environment::Development,
|
|
||||||
_ => Environment::Production,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn keyring_entry(environment: Environment) -> CommandResult<Entry> {
|
|
||||||
Entry::new(environment.keyring_service(), KEYRING_USER)
|
|
||||||
.map_err(|e| format!("Failed to initialize auth keyring entry: {e}"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_auth_token(environment: Environment) -> CommandResult<Option<String>> {
|
|
||||||
let entry = keyring_entry(environment)?;
|
|
||||||
match entry.get_password() {
|
|
||||||
Ok(token) => Ok(Some(token)),
|
|
||||||
Err(keyring::Error::NoEntry) => Ok(None),
|
|
||||||
Err(err) => Err(format!("Failed to read auth token: {err}")),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn random_name() -> String {
|
|
||||||
const ADJECTIVES: &[&str] = &[
|
|
||||||
"young", "youthful", "yellow", "yielding", "yappy", "yawning", "yummy", "yucky", "yearly",
|
|
||||||
"yester", "yeasty", "yelling",
|
|
||||||
];
|
|
||||||
const NOUNS: &[&str] = &[
|
|
||||||
"yak", "yarn", "year", "yell", "yoke", "yoga", "yam", "yacht", "yodel",
|
|
||||||
];
|
|
||||||
|
|
||||||
let mut rng = rand::thread_rng();
|
|
||||||
let adjective = ADJECTIVES[rng.gen_range(0..ADJECTIVES.len())];
|
|
||||||
let noun = NOUNS[rng.gen_range(0..NOUNS.len())];
|
|
||||||
format!("{adjective}-{noun}")
|
|
||||||
}
|
|
||||||
|
|
||||||
const TEMPLATE_GITIGNORE: &str = "node_modules\n";
|
|
||||||
|
|
||||||
const TEMPLATE_PACKAGE_JSON: &str = r#"{
|
|
||||||
"name": "yaak-plugin-name",
|
|
||||||
"private": true,
|
|
||||||
"version": "0.0.1",
|
|
||||||
"scripts": {
|
|
||||||
"build": "yaak plugin build",
|
|
||||||
"dev": "yaak plugin dev"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@types/node": "^24.10.1",
|
|
||||||
"typescript": "^5.9.3",
|
|
||||||
"vitest": "^4.0.14"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"@yaakapp/api": "^0.7.0"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"#;
|
|
||||||
|
|
||||||
const TEMPLATE_TSCONFIG: &str = r#"{
|
|
||||||
"compilerOptions": {
|
|
||||||
"target": "es2021",
|
|
||||||
"lib": ["DOM", "DOM.Iterable", "ESNext"],
|
|
||||||
"useDefineForClassFields": true,
|
|
||||||
"allowJs": false,
|
|
||||||
"skipLibCheck": true,
|
|
||||||
"esModuleInterop": false,
|
|
||||||
"allowSyntheticDefaultImports": true,
|
|
||||||
"strict": true,
|
|
||||||
"noUncheckedIndexedAccess": true,
|
|
||||||
"forceConsistentCasingInFileNames": true,
|
|
||||||
"module": "ESNext",
|
|
||||||
"moduleResolution": "Node",
|
|
||||||
"resolveJsonModule": true,
|
|
||||||
"isolatedModules": true,
|
|
||||||
"noEmit": true,
|
|
||||||
"jsx": "react-jsx"
|
|
||||||
},
|
|
||||||
"include": ["src"]
|
|
||||||
}
|
|
||||||
"#;
|
|
||||||
|
|
||||||
const TEMPLATE_README: &str = r#"# yaak-plugin-name
|
|
||||||
|
|
||||||
Describe what your plugin does.
|
|
||||||
"#;
|
|
||||||
|
|
||||||
const TEMPLATE_INDEX_TS: &str = r#"import type { PluginDefinition } from "@yaakapp/api";
|
|
||||||
|
|
||||||
export const plugin: PluginDefinition = {
|
|
||||||
httpRequestActions: [
|
|
||||||
{
|
|
||||||
label: "Hello, From Plugin",
|
|
||||||
icon: "info",
|
|
||||||
async onSelect(ctx, args) {
|
|
||||||
await ctx.toast.show({
|
|
||||||
color: "success",
|
|
||||||
message: `You clicked the request ${args.httpRequest.id}`,
|
|
||||||
});
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
|
||||||
"#;
|
|
||||||
|
|
||||||
const TEMPLATE_INDEX_TEST_TS: &str = r#"import { describe, expect, test } from "vitest";
|
|
||||||
import { plugin } from "./index";
|
|
||||||
|
|
||||||
describe("Example Plugin", () => {
|
|
||||||
test("Exports plugin object", () => {
|
|
||||||
expect(plugin).toBeTypeOf("object");
|
|
||||||
});
|
|
||||||
});
|
|
||||||
"#;
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod tests {
|
|
||||||
use super::create_publish_archive;
|
|
||||||
use std::collections::HashSet;
|
|
||||||
use std::fs;
|
|
||||||
use std::io::Cursor;
|
|
||||||
use tempfile::TempDir;
|
|
||||||
use zip::ZipArchive;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn publish_archive_includes_required_and_optional_files() {
|
|
||||||
let dir = TempDir::new().expect("temp dir");
|
|
||||||
let root = dir.path();
|
|
||||||
|
|
||||||
fs::create_dir_all(root.join("src")).expect("create src");
|
|
||||||
fs::create_dir_all(root.join("build")).expect("create build");
|
|
||||||
fs::create_dir_all(root.join("ignored")).expect("create ignored");
|
|
||||||
|
|
||||||
fs::write(root.join("README.md"), "# Demo\n").expect("write README");
|
|
||||||
fs::write(root.join("package.json"), "{}").expect("write package.json");
|
|
||||||
fs::write(root.join("package-lock.json"), "{}").expect("write package-lock.json");
|
|
||||||
fs::write(root.join("src/index.ts"), "export const plugin = {};\n")
|
|
||||||
.expect("write src/index.ts");
|
|
||||||
fs::write(root.join("build/index.js"), "exports.plugin = {};\n")
|
|
||||||
.expect("write build/index.js");
|
|
||||||
fs::write(root.join("ignored/secret.txt"), "do-not-ship").expect("write ignored file");
|
|
||||||
|
|
||||||
let archive = create_publish_archive(root).expect("create archive");
|
|
||||||
let mut zip = ZipArchive::new(Cursor::new(archive)).expect("open zip");
|
|
||||||
|
|
||||||
let mut names = HashSet::new();
|
|
||||||
for i in 0..zip.len() {
|
|
||||||
let file = zip.by_index(i).expect("zip entry");
|
|
||||||
names.insert(file.name().to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
assert!(names.contains("README.md"));
|
|
||||||
assert!(names.contains("package.json"));
|
|
||||||
assert!(names.contains("package-lock.json"));
|
|
||||||
assert!(names.contains("src/index.ts"));
|
|
||||||
assert!(names.contains("build/index.js"));
|
|
||||||
assert!(!names.contains("ignored/secret.txt"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,514 +0,0 @@
|
|||||||
use crate::cli::{RequestArgs, RequestCommands, RequestSchemaType};
|
|
||||||
use crate::context::CliContext;
|
|
||||||
use crate::utils::confirm::confirm_delete;
|
|
||||||
use crate::utils::json::{
|
|
||||||
apply_merge_patch, is_json_shorthand, merge_workspace_id_arg, parse_optional_json,
|
|
||||||
parse_required_json, require_id, validate_create_id,
|
|
||||||
};
|
|
||||||
use crate::utils::schema::append_agent_hints;
|
|
||||||
use schemars::schema_for;
|
|
||||||
use serde_json::{Map, Value, json};
|
|
||||||
use std::collections::HashMap;
|
|
||||||
use std::io::Write;
|
|
||||||
use tokio::sync::mpsc;
|
|
||||||
use yaak::send::{SendHttpRequestByIdWithPluginsParams, send_http_request_by_id_with_plugins};
|
|
||||||
use yaak_http::sender::HttpResponseEvent as SenderHttpResponseEvent;
|
|
||||||
use yaak_models::models::{GrpcRequest, HttpRequest, WebsocketRequest};
|
|
||||||
use yaak_models::queries::any_request::AnyRequest;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
use yaak_plugins::events::{FormInput, FormInputBase, JsonPrimitive, PluginContext};
|
|
||||||
|
|
||||||
type CommandResult<T = ()> = std::result::Result<T, String>;
|
|
||||||
|
|
||||||
pub async fn run(
|
|
||||||
ctx: &CliContext,
|
|
||||||
args: RequestArgs,
|
|
||||||
environment: Option<&str>,
|
|
||||||
verbose: bool,
|
|
||||||
) -> i32 {
|
|
||||||
let result = match args.command {
|
|
||||||
RequestCommands::List { workspace_id } => list(ctx, &workspace_id),
|
|
||||||
RequestCommands::Show { request_id } => show(ctx, &request_id),
|
|
||||||
RequestCommands::Send { request_id } => {
|
|
||||||
return match send_request_by_id(ctx, &request_id, environment, verbose).await {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!("Error: {error}");
|
|
||||||
1
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
RequestCommands::Schema { request_type, pretty } => {
|
|
||||||
return match schema(ctx, request_type, pretty).await {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!("Error: {error}");
|
|
||||||
1
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
RequestCommands::Create { workspace_id, name, method, url, json } => {
|
|
||||||
create(ctx, workspace_id, name, method, url, json)
|
|
||||||
}
|
|
||||||
RequestCommands::Update { json, json_input } => update(ctx, json, json_input),
|
|
||||||
RequestCommands::Delete { request_id, yes } => delete(ctx, &request_id, yes),
|
|
||||||
};
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!("Error: {error}");
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn list(ctx: &CliContext, workspace_id: &str) -> CommandResult {
|
|
||||||
let requests = ctx
|
|
||||||
.db()
|
|
||||||
.list_http_requests(workspace_id)
|
|
||||||
.map_err(|e| format!("Failed to list requests: {e}"))?;
|
|
||||||
if requests.is_empty() {
|
|
||||||
println!("No requests found in workspace {}", workspace_id);
|
|
||||||
} else {
|
|
||||||
for request in requests {
|
|
||||||
println!("{} - {} {}", request.id, request.method, request.name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn schema(ctx: &CliContext, request_type: RequestSchemaType, pretty: bool) -> CommandResult {
|
|
||||||
let mut schema = match request_type {
|
|
||||||
RequestSchemaType::Http => serde_json::to_value(schema_for!(HttpRequest))
|
|
||||||
.map_err(|e| format!("Failed to serialize HTTP request schema: {e}"))?,
|
|
||||||
RequestSchemaType::Grpc => serde_json::to_value(schema_for!(GrpcRequest))
|
|
||||||
.map_err(|e| format!("Failed to serialize gRPC request schema: {e}"))?,
|
|
||||||
RequestSchemaType::Websocket => serde_json::to_value(schema_for!(WebsocketRequest))
|
|
||||||
.map_err(|e| format!("Failed to serialize WebSocket request schema: {e}"))?,
|
|
||||||
};
|
|
||||||
|
|
||||||
enrich_schema_guidance(&mut schema, request_type);
|
|
||||||
append_agent_hints(&mut schema);
|
|
||||||
|
|
||||||
if let Err(error) = merge_auth_schema_from_plugins(ctx, &mut schema).await {
|
|
||||||
eprintln!("Warning: Failed to enrich authentication schema from plugins: {error}");
|
|
||||||
}
|
|
||||||
|
|
||||||
let output =
|
|
||||||
if pretty { serde_json::to_string_pretty(&schema) } else { serde_json::to_string(&schema) }
|
|
||||||
.map_err(|e| format!("Failed to format schema JSON: {e}"))?;
|
|
||||||
println!("{output}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn enrich_schema_guidance(schema: &mut Value, request_type: RequestSchemaType) {
|
|
||||||
if !matches!(request_type, RequestSchemaType::Http) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let Some(properties) = schema.get_mut("properties").and_then(Value::as_object_mut) else {
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(url_schema) = properties.get_mut("url").and_then(Value::as_object_mut) {
|
|
||||||
append_description(
|
|
||||||
url_schema,
|
|
||||||
"For path segments like `/foo/:id/comments/:commentId`, put concrete values in `urlParameters` using names without `:` (for example `id`, `commentId`).",
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn append_description(schema: &mut Map<String, Value>, extra: &str) {
|
|
||||||
match schema.get_mut("description") {
|
|
||||||
Some(Value::String(existing)) if !existing.trim().is_empty() => {
|
|
||||||
if !existing.ends_with(' ') {
|
|
||||||
existing.push(' ');
|
|
||||||
}
|
|
||||||
existing.push_str(extra);
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
schema.insert("description".to_string(), Value::String(extra.to_string()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn merge_auth_schema_from_plugins(
|
|
||||||
ctx: &CliContext,
|
|
||||||
schema: &mut Value,
|
|
||||||
) -> Result<(), String> {
|
|
||||||
let plugin_context = PluginContext::new_empty();
|
|
||||||
let plugin_manager = ctx.plugin_manager();
|
|
||||||
let summaries = plugin_manager
|
|
||||||
.get_http_authentication_summaries(&plugin_context)
|
|
||||||
.await
|
|
||||||
.map_err(|e| e.to_string())?;
|
|
||||||
|
|
||||||
let mut auth_variants = Vec::new();
|
|
||||||
for (_, summary) in summaries {
|
|
||||||
let config = match plugin_manager
|
|
||||||
.get_http_authentication_config(
|
|
||||||
&plugin_context,
|
|
||||||
&summary.name,
|
|
||||||
HashMap::<String, JsonPrimitive>::new(),
|
|
||||||
"yaakcli_request_schema",
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
Ok(config) => config,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!(
|
|
||||||
"Warning: Failed to load auth config for strategy '{}': {}",
|
|
||||||
summary.name, error
|
|
||||||
);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
auth_variants.push(auth_variant_schema(&summary.name, &summary.label, &config.args));
|
|
||||||
}
|
|
||||||
|
|
||||||
let Some(properties) = schema.get_mut("properties").and_then(Value::as_object_mut) else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
let Some(auth_schema) = properties.get_mut("authentication") else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
if !auth_variants.is_empty() {
|
|
||||||
let mut one_of = vec![auth_schema.clone()];
|
|
||||||
one_of.extend(auth_variants);
|
|
||||||
*auth_schema = json!({ "oneOf": one_of });
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn auth_variant_schema(auth_name: &str, auth_label: &str, args: &[FormInput]) -> Value {
|
|
||||||
let mut properties = Map::new();
|
|
||||||
let mut required = Vec::new();
|
|
||||||
for input in args {
|
|
||||||
add_input_schema(input, &mut properties, &mut required);
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut schema = json!({
|
|
||||||
"title": auth_label,
|
|
||||||
"description": format!("Authentication values for strategy '{}'", auth_name),
|
|
||||||
"type": "object",
|
|
||||||
"properties": properties,
|
|
||||||
"additionalProperties": true
|
|
||||||
});
|
|
||||||
|
|
||||||
if !required.is_empty() {
|
|
||||||
schema["required"] = json!(required);
|
|
||||||
}
|
|
||||||
|
|
||||||
schema
|
|
||||||
}
|
|
||||||
|
|
||||||
fn add_input_schema(
|
|
||||||
input: &FormInput,
|
|
||||||
properties: &mut Map<String, Value>,
|
|
||||||
required: &mut Vec<String>,
|
|
||||||
) {
|
|
||||||
match input {
|
|
||||||
FormInput::Text(v) => add_base_schema(
|
|
||||||
&v.base,
|
|
||||||
json!({
|
|
||||||
"type": "string",
|
|
||||||
"writeOnly": v.password.unwrap_or(false),
|
|
||||||
}),
|
|
||||||
properties,
|
|
||||||
required,
|
|
||||||
),
|
|
||||||
FormInput::Editor(v) => add_base_schema(
|
|
||||||
&v.base,
|
|
||||||
json!({
|
|
||||||
"type": "string",
|
|
||||||
"x-editorLanguage": v.language.clone(),
|
|
||||||
}),
|
|
||||||
properties,
|
|
||||||
required,
|
|
||||||
),
|
|
||||||
FormInput::Select(v) => {
|
|
||||||
let options: Vec<Value> =
|
|
||||||
v.options.iter().map(|o| Value::String(o.value.clone())).collect();
|
|
||||||
add_base_schema(
|
|
||||||
&v.base,
|
|
||||||
json!({
|
|
||||||
"type": "string",
|
|
||||||
"enum": options,
|
|
||||||
}),
|
|
||||||
properties,
|
|
||||||
required,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
FormInput::Checkbox(v) => {
|
|
||||||
add_base_schema(&v.base, json!({ "type": "boolean" }), properties, required);
|
|
||||||
}
|
|
||||||
FormInput::File(v) => {
|
|
||||||
if v.multiple.unwrap_or(false) {
|
|
||||||
add_base_schema(
|
|
||||||
&v.base,
|
|
||||||
json!({
|
|
||||||
"type": "array",
|
|
||||||
"items": { "type": "string" },
|
|
||||||
}),
|
|
||||||
properties,
|
|
||||||
required,
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
add_base_schema(&v.base, json!({ "type": "string" }), properties, required);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FormInput::HttpRequest(v) => {
|
|
||||||
add_base_schema(&v.base, json!({ "type": "string" }), properties, required);
|
|
||||||
}
|
|
||||||
FormInput::KeyValue(v) => {
|
|
||||||
add_base_schema(
|
|
||||||
&v.base,
|
|
||||||
json!({
|
|
||||||
"type": "object",
|
|
||||||
"additionalProperties": true,
|
|
||||||
}),
|
|
||||||
properties,
|
|
||||||
required,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
FormInput::Accordion(v) => {
|
|
||||||
if let Some(children) = &v.inputs {
|
|
||||||
for child in children {
|
|
||||||
add_input_schema(child, properties, required);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FormInput::HStack(v) => {
|
|
||||||
if let Some(children) = &v.inputs {
|
|
||||||
for child in children {
|
|
||||||
add_input_schema(child, properties, required);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FormInput::Banner(v) => {
|
|
||||||
if let Some(children) = &v.inputs {
|
|
||||||
for child in children {
|
|
||||||
add_input_schema(child, properties, required);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FormInput::Markdown(_) => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn add_base_schema(
|
|
||||||
base: &FormInputBase,
|
|
||||||
mut schema: Value,
|
|
||||||
properties: &mut Map<String, Value>,
|
|
||||||
required: &mut Vec<String>,
|
|
||||||
) {
|
|
||||||
if base.hidden.unwrap_or(false) || base.name.trim().is_empty() {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(description) = &base.description {
|
|
||||||
schema["description"] = Value::String(description.clone());
|
|
||||||
}
|
|
||||||
if let Some(label) = &base.label {
|
|
||||||
schema["title"] = Value::String(label.clone());
|
|
||||||
}
|
|
||||||
if let Some(default_value) = &base.default_value {
|
|
||||||
schema["default"] = Value::String(default_value.clone());
|
|
||||||
}
|
|
||||||
|
|
||||||
let name = base.name.clone();
|
|
||||||
properties.insert(name.clone(), schema);
|
|
||||||
if !base.optional.unwrap_or(false) {
|
|
||||||
required.push(name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn create(
|
|
||||||
ctx: &CliContext,
|
|
||||||
workspace_id: Option<String>,
|
|
||||||
name: Option<String>,
|
|
||||||
method: Option<String>,
|
|
||||||
url: Option<String>,
|
|
||||||
json: Option<String>,
|
|
||||||
) -> CommandResult {
|
|
||||||
let json_shorthand =
|
|
||||||
workspace_id.as_deref().filter(|v| is_json_shorthand(v)).map(str::to_owned);
|
|
||||||
let workspace_id_arg = workspace_id.filter(|v| !is_json_shorthand(v));
|
|
||||||
|
|
||||||
let payload = parse_optional_json(json, json_shorthand, "request create")?;
|
|
||||||
|
|
||||||
if let Some(payload) = payload {
|
|
||||||
if name.is_some() || method.is_some() || url.is_some() {
|
|
||||||
return Err("request create cannot combine simple flags with JSON payload".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
validate_create_id(&payload, "request")?;
|
|
||||||
let mut request: HttpRequest = serde_json::from_value(payload)
|
|
||||||
.map_err(|e| format!("Failed to parse request create JSON: {e}"))?;
|
|
||||||
merge_workspace_id_arg(
|
|
||||||
workspace_id_arg.as_deref(),
|
|
||||||
&mut request.workspace_id,
|
|
||||||
"request create",
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_http_request(&request, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create request: {e}"))?;
|
|
||||||
|
|
||||||
println!("Created request: {}", created.id);
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let workspace_id = workspace_id_arg.ok_or_else(|| {
|
|
||||||
"request create requires workspace_id unless JSON payload is provided".to_string()
|
|
||||||
})?;
|
|
||||||
let name = name.unwrap_or_default();
|
|
||||||
let url = url.unwrap_or_default();
|
|
||||||
let method = method.unwrap_or_else(|| "GET".to_string());
|
|
||||||
|
|
||||||
let request = HttpRequest {
|
|
||||||
workspace_id,
|
|
||||||
name,
|
|
||||||
method: method.to_uppercase(),
|
|
||||||
url,
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_http_request(&request, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create request: {e}"))?;
|
|
||||||
|
|
||||||
println!("Created request: {}", created.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update(ctx: &CliContext, json: Option<String>, json_input: Option<String>) -> CommandResult {
|
|
||||||
let patch = parse_required_json(json, json_input, "request update")?;
|
|
||||||
let id = require_id(&patch, "request update")?;
|
|
||||||
|
|
||||||
let existing = ctx
|
|
||||||
.db()
|
|
||||||
.get_http_request(&id)
|
|
||||||
.map_err(|e| format!("Failed to get request for update: {e}"))?;
|
|
||||||
let updated = apply_merge_patch(&existing, &patch, &id, "request update")?;
|
|
||||||
|
|
||||||
let saved = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_http_request(&updated, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to update request: {e}"))?;
|
|
||||||
|
|
||||||
println!("Updated request: {}", saved.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn show(ctx: &CliContext, request_id: &str) -> CommandResult {
|
|
||||||
let request =
|
|
||||||
ctx.db().get_http_request(request_id).map_err(|e| format!("Failed to get request: {e}"))?;
|
|
||||||
let output = serde_json::to_string_pretty(&request)
|
|
||||||
.map_err(|e| format!("Failed to serialize request: {e}"))?;
|
|
||||||
println!("{output}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn delete(ctx: &CliContext, request_id: &str, yes: bool) -> CommandResult {
|
|
||||||
if !yes && !confirm_delete("request", request_id) {
|
|
||||||
println!("Aborted");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let deleted = ctx
|
|
||||||
.db()
|
|
||||||
.delete_http_request_by_id(request_id, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to delete request: {e}"))?;
|
|
||||||
println!("Deleted request: {}", deleted.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Send a request by ID and print response in the same format as legacy `send`.
|
|
||||||
pub async fn send_request_by_id(
|
|
||||||
ctx: &CliContext,
|
|
||||||
request_id: &str,
|
|
||||||
environment: Option<&str>,
|
|
||||||
verbose: bool,
|
|
||||||
) -> Result<(), String> {
|
|
||||||
let request =
|
|
||||||
ctx.db().get_any_request(request_id).map_err(|e| format!("Failed to get request: {e}"))?;
|
|
||||||
match request {
|
|
||||||
AnyRequest::HttpRequest(http_request) => {
|
|
||||||
send_http_request_by_id(
|
|
||||||
ctx,
|
|
||||||
&http_request.id,
|
|
||||||
&http_request.workspace_id,
|
|
||||||
environment,
|
|
||||||
verbose,
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
AnyRequest::GrpcRequest(_) => {
|
|
||||||
Err("gRPC request send is not implemented yet in yaak-cli".to_string())
|
|
||||||
}
|
|
||||||
AnyRequest::WebsocketRequest(_) => {
|
|
||||||
Err("WebSocket request send is not implemented yet in yaak-cli".to_string())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn send_http_request_by_id(
|
|
||||||
ctx: &CliContext,
|
|
||||||
request_id: &str,
|
|
||||||
workspace_id: &str,
|
|
||||||
environment: Option<&str>,
|
|
||||||
verbose: bool,
|
|
||||||
) -> Result<(), String> {
|
|
||||||
let plugin_context = PluginContext::new(None, Some(workspace_id.to_string()));
|
|
||||||
|
|
||||||
let (event_tx, mut event_rx) = mpsc::channel::<SenderHttpResponseEvent>(100);
|
|
||||||
let (body_chunk_tx, mut body_chunk_rx) = mpsc::unbounded_channel::<Vec<u8>>();
|
|
||||||
let event_handle = tokio::spawn(async move {
|
|
||||||
while let Some(event) = event_rx.recv().await {
|
|
||||||
if verbose && !matches!(event, SenderHttpResponseEvent::ChunkReceived { .. }) {
|
|
||||||
println!("{}", event);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
let body_handle = tokio::task::spawn_blocking(move || {
|
|
||||||
let mut stdout = std::io::stdout();
|
|
||||||
while let Some(chunk) = body_chunk_rx.blocking_recv() {
|
|
||||||
if stdout.write_all(&chunk).is_err() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
let _ = stdout.flush();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
let response_dir = ctx.data_dir().join("responses");
|
|
||||||
|
|
||||||
let result = send_http_request_by_id_with_plugins(SendHttpRequestByIdWithPluginsParams {
|
|
||||||
query_manager: ctx.query_manager(),
|
|
||||||
blob_manager: ctx.blob_manager(),
|
|
||||||
request_id,
|
|
||||||
environment_id: environment,
|
|
||||||
update_source: UpdateSource::Sync,
|
|
||||||
cookie_jar_id: None,
|
|
||||||
response_dir: &response_dir,
|
|
||||||
emit_events_to: Some(event_tx),
|
|
||||||
emit_response_body_chunks_to: Some(body_chunk_tx),
|
|
||||||
plugin_manager: ctx.plugin_manager(),
|
|
||||||
encryption_manager: ctx.encryption_manager.clone(),
|
|
||||||
plugin_context: &plugin_context,
|
|
||||||
cancelled_rx: None,
|
|
||||||
connection_manager: None,
|
|
||||||
})
|
|
||||||
.await;
|
|
||||||
|
|
||||||
let _ = event_handle.await;
|
|
||||||
let _ = body_handle.await;
|
|
||||||
result.map_err(|e| e.to_string())?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
@@ -1,184 +0,0 @@
|
|||||||
use crate::cli::SendArgs;
|
|
||||||
use crate::commands::request;
|
|
||||||
use crate::context::CliContext;
|
|
||||||
use futures::future::join_all;
|
|
||||||
|
|
||||||
enum ExecutionMode {
|
|
||||||
Sequential,
|
|
||||||
Parallel,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn run(
|
|
||||||
ctx: &CliContext,
|
|
||||||
args: SendArgs,
|
|
||||||
environment: Option<&str>,
|
|
||||||
verbose: bool,
|
|
||||||
) -> i32 {
|
|
||||||
match send_target(ctx, args, environment, verbose).await {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!("Error: {error}");
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn send_target(
|
|
||||||
ctx: &CliContext,
|
|
||||||
args: SendArgs,
|
|
||||||
environment: Option<&str>,
|
|
||||||
verbose: bool,
|
|
||||||
) -> Result<(), String> {
|
|
||||||
let mode = if args.parallel { ExecutionMode::Parallel } else { ExecutionMode::Sequential };
|
|
||||||
|
|
||||||
if ctx.db().get_any_request(&args.id).is_ok() {
|
|
||||||
return request::send_request_by_id(ctx, &args.id, environment, verbose).await;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ctx.db().get_folder(&args.id).is_ok() {
|
|
||||||
let request_ids = collect_folder_request_ids(ctx, &args.id)?;
|
|
||||||
if request_ids.is_empty() {
|
|
||||||
println!("No requests found in folder {}", args.id);
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
return send_many(ctx, request_ids, mode, args.fail_fast, environment, verbose).await;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ctx.db().get_workspace(&args.id).is_ok() {
|
|
||||||
let request_ids = collect_workspace_request_ids(ctx, &args.id)?;
|
|
||||||
if request_ids.is_empty() {
|
|
||||||
println!("No requests found in workspace {}", args.id);
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
return send_many(ctx, request_ids, mode, args.fail_fast, environment, verbose).await;
|
|
||||||
}
|
|
||||||
|
|
||||||
Err(format!("Could not resolve ID '{}' as request, folder, or workspace", args.id))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn collect_folder_request_ids(ctx: &CliContext, folder_id: &str) -> Result<Vec<String>, String> {
|
|
||||||
let mut ids = Vec::new();
|
|
||||||
|
|
||||||
let mut http_ids = ctx
|
|
||||||
.db()
|
|
||||||
.list_http_requests_for_folder_recursive(folder_id)
|
|
||||||
.map_err(|e| format!("Failed to list HTTP requests in folder: {e}"))?
|
|
||||||
.into_iter()
|
|
||||||
.map(|r| r.id)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
ids.append(&mut http_ids);
|
|
||||||
|
|
||||||
let mut grpc_ids = ctx
|
|
||||||
.db()
|
|
||||||
.list_grpc_requests_for_folder_recursive(folder_id)
|
|
||||||
.map_err(|e| format!("Failed to list gRPC requests in folder: {e}"))?
|
|
||||||
.into_iter()
|
|
||||||
.map(|r| r.id)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
ids.append(&mut grpc_ids);
|
|
||||||
|
|
||||||
let mut websocket_ids = ctx
|
|
||||||
.db()
|
|
||||||
.list_websocket_requests_for_folder_recursive(folder_id)
|
|
||||||
.map_err(|e| format!("Failed to list WebSocket requests in folder: {e}"))?
|
|
||||||
.into_iter()
|
|
||||||
.map(|r| r.id)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
ids.append(&mut websocket_ids);
|
|
||||||
|
|
||||||
Ok(ids)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn collect_workspace_request_ids(
|
|
||||||
ctx: &CliContext,
|
|
||||||
workspace_id: &str,
|
|
||||||
) -> Result<Vec<String>, String> {
|
|
||||||
let mut ids = Vec::new();
|
|
||||||
|
|
||||||
let mut http_ids = ctx
|
|
||||||
.db()
|
|
||||||
.list_http_requests(workspace_id)
|
|
||||||
.map_err(|e| format!("Failed to list HTTP requests in workspace: {e}"))?
|
|
||||||
.into_iter()
|
|
||||||
.map(|r| r.id)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
ids.append(&mut http_ids);
|
|
||||||
|
|
||||||
let mut grpc_ids = ctx
|
|
||||||
.db()
|
|
||||||
.list_grpc_requests(workspace_id)
|
|
||||||
.map_err(|e| format!("Failed to list gRPC requests in workspace: {e}"))?
|
|
||||||
.into_iter()
|
|
||||||
.map(|r| r.id)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
ids.append(&mut grpc_ids);
|
|
||||||
|
|
||||||
let mut websocket_ids = ctx
|
|
||||||
.db()
|
|
||||||
.list_websocket_requests(workspace_id)
|
|
||||||
.map_err(|e| format!("Failed to list WebSocket requests in workspace: {e}"))?
|
|
||||||
.into_iter()
|
|
||||||
.map(|r| r.id)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
ids.append(&mut websocket_ids);
|
|
||||||
|
|
||||||
Ok(ids)
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn send_many(
|
|
||||||
ctx: &CliContext,
|
|
||||||
request_ids: Vec<String>,
|
|
||||||
mode: ExecutionMode,
|
|
||||||
fail_fast: bool,
|
|
||||||
environment: Option<&str>,
|
|
||||||
verbose: bool,
|
|
||||||
) -> Result<(), String> {
|
|
||||||
let mut success_count = 0usize;
|
|
||||||
let mut failures: Vec<(String, String)> = Vec::new();
|
|
||||||
|
|
||||||
match mode {
|
|
||||||
ExecutionMode::Sequential => {
|
|
||||||
for request_id in request_ids {
|
|
||||||
match request::send_request_by_id(ctx, &request_id, environment, verbose).await {
|
|
||||||
Ok(()) => success_count += 1,
|
|
||||||
Err(error) => {
|
|
||||||
failures.push((request_id, error));
|
|
||||||
if fail_fast {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ExecutionMode::Parallel => {
|
|
||||||
let tasks = request_ids
|
|
||||||
.iter()
|
|
||||||
.map(|request_id| async move {
|
|
||||||
(
|
|
||||||
request_id.clone(),
|
|
||||||
request::send_request_by_id(ctx, request_id, environment, verbose).await,
|
|
||||||
)
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
for (request_id, result) in join_all(tasks).await {
|
|
||||||
match result {
|
|
||||||
Ok(()) => success_count += 1,
|
|
||||||
Err(error) => failures.push((request_id, error)),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let failure_count = failures.len();
|
|
||||||
println!("Send summary: {success_count} succeeded, {failure_count} failed");
|
|
||||||
|
|
||||||
if failure_count == 0 {
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
for (request_id, error) in failures {
|
|
||||||
eprintln!(" {}: {}", request_id, error);
|
|
||||||
}
|
|
||||||
Err("One or more requests failed".to_string())
|
|
||||||
}
|
|
||||||
@@ -1,143 +0,0 @@
|
|||||||
use crate::cli::{WorkspaceArgs, WorkspaceCommands};
|
|
||||||
use crate::context::CliContext;
|
|
||||||
use crate::utils::confirm::confirm_delete;
|
|
||||||
use crate::utils::json::{
|
|
||||||
apply_merge_patch, parse_optional_json, parse_required_json, require_id, validate_create_id,
|
|
||||||
};
|
|
||||||
use crate::utils::schema::append_agent_hints;
|
|
||||||
use schemars::schema_for;
|
|
||||||
use yaak_models::models::Workspace;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
type CommandResult<T = ()> = std::result::Result<T, String>;
|
|
||||||
|
|
||||||
pub fn run(ctx: &CliContext, args: WorkspaceArgs) -> i32 {
|
|
||||||
let result = match args.command {
|
|
||||||
WorkspaceCommands::List => list(ctx),
|
|
||||||
WorkspaceCommands::Schema { pretty } => schema(pretty),
|
|
||||||
WorkspaceCommands::Show { workspace_id } => show(ctx, &workspace_id),
|
|
||||||
WorkspaceCommands::Create { name, json, json_input } => create(ctx, name, json, json_input),
|
|
||||||
WorkspaceCommands::Update { json, json_input } => update(ctx, json, json_input),
|
|
||||||
WorkspaceCommands::Delete { workspace_id, yes } => delete(ctx, &workspace_id, yes),
|
|
||||||
};
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(()) => 0,
|
|
||||||
Err(error) => {
|
|
||||||
eprintln!("Error: {error}");
|
|
||||||
1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn schema(pretty: bool) -> CommandResult {
|
|
||||||
let mut schema =
|
|
||||||
serde_json::to_value(schema_for!(Workspace)).map_err(|e| format!(
|
|
||||||
"Failed to serialize workspace schema: {e}"
|
|
||||||
))?;
|
|
||||||
append_agent_hints(&mut schema);
|
|
||||||
|
|
||||||
let output = if pretty {
|
|
||||||
serde_json::to_string_pretty(&schema)
|
|
||||||
} else {
|
|
||||||
serde_json::to_string(&schema)
|
|
||||||
}
|
|
||||||
.map_err(|e| format!("Failed to format workspace schema JSON: {e}"))?;
|
|
||||||
println!("{output}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn list(ctx: &CliContext) -> CommandResult {
|
|
||||||
let workspaces =
|
|
||||||
ctx.db().list_workspaces().map_err(|e| format!("Failed to list workspaces: {e}"))?;
|
|
||||||
if workspaces.is_empty() {
|
|
||||||
println!("No workspaces found");
|
|
||||||
} else {
|
|
||||||
for workspace in workspaces {
|
|
||||||
println!("{} - {}", workspace.id, workspace.name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn show(ctx: &CliContext, workspace_id: &str) -> CommandResult {
|
|
||||||
let workspace = ctx
|
|
||||||
.db()
|
|
||||||
.get_workspace(workspace_id)
|
|
||||||
.map_err(|e| format!("Failed to get workspace: {e}"))?;
|
|
||||||
let output = serde_json::to_string_pretty(&workspace)
|
|
||||||
.map_err(|e| format!("Failed to serialize workspace: {e}"))?;
|
|
||||||
println!("{output}");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn create(
|
|
||||||
ctx: &CliContext,
|
|
||||||
name: Option<String>,
|
|
||||||
json: Option<String>,
|
|
||||||
json_input: Option<String>,
|
|
||||||
) -> CommandResult {
|
|
||||||
let payload = parse_optional_json(json, json_input, "workspace create")?;
|
|
||||||
|
|
||||||
if let Some(payload) = payload {
|
|
||||||
if name.is_some() {
|
|
||||||
return Err("workspace create cannot combine --name with JSON payload".to_string());
|
|
||||||
}
|
|
||||||
|
|
||||||
validate_create_id(&payload, "workspace")?;
|
|
||||||
let workspace: Workspace = serde_json::from_value(payload)
|
|
||||||
.map_err(|e| format!("Failed to parse workspace create JSON: {e}"))?;
|
|
||||||
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_workspace(&workspace, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create workspace: {e}"))?;
|
|
||||||
println!("Created workspace: {}", created.id);
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let name = name.ok_or_else(|| {
|
|
||||||
"workspace create requires --name unless JSON payload is provided".to_string()
|
|
||||||
})?;
|
|
||||||
|
|
||||||
let workspace = Workspace { name, ..Default::default() };
|
|
||||||
let created = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_workspace(&workspace, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to create workspace: {e}"))?;
|
|
||||||
println!("Created workspace: {}", created.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update(ctx: &CliContext, json: Option<String>, json_input: Option<String>) -> CommandResult {
|
|
||||||
let patch = parse_required_json(json, json_input, "workspace update")?;
|
|
||||||
let id = require_id(&patch, "workspace update")?;
|
|
||||||
|
|
||||||
let existing = ctx
|
|
||||||
.db()
|
|
||||||
.get_workspace(&id)
|
|
||||||
.map_err(|e| format!("Failed to get workspace for update: {e}"))?;
|
|
||||||
let updated = apply_merge_patch(&existing, &patch, &id, "workspace update")?;
|
|
||||||
|
|
||||||
let saved = ctx
|
|
||||||
.db()
|
|
||||||
.upsert_workspace(&updated, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to update workspace: {e}"))?;
|
|
||||||
|
|
||||||
println!("Updated workspace: {}", saved.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn delete(ctx: &CliContext, workspace_id: &str, yes: bool) -> CommandResult {
|
|
||||||
if !yes && !confirm_delete("workspace", workspace_id) {
|
|
||||||
println!("Aborted");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let deleted = ctx
|
|
||||||
.db()
|
|
||||||
.delete_workspace_by_id(workspace_id, &UpdateSource::Sync)
|
|
||||||
.map_err(|e| format!("Failed to delete workspace: {e}"))?;
|
|
||||||
println!("Deleted workspace: {}", deleted.id);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
@@ -1,154 +0,0 @@
|
|||||||
use crate::plugin_events::CliPluginEventBridge;
|
|
||||||
use include_dir::{Dir, include_dir};
|
|
||||||
use std::fs;
|
|
||||||
use std::path::{Path, PathBuf};
|
|
||||||
use std::sync::Arc;
|
|
||||||
use tokio::sync::Mutex;
|
|
||||||
use yaak_crypto::manager::EncryptionManager;
|
|
||||||
use yaak_models::blob_manager::BlobManager;
|
|
||||||
use yaak_models::db_context::DbContext;
|
|
||||||
use yaak_models::query_manager::QueryManager;
|
|
||||||
use yaak_plugins::events::PluginContext;
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
|
|
||||||
const EMBEDDED_PLUGIN_RUNTIME: &str = include_str!(concat!(
|
|
||||||
env!("CARGO_MANIFEST_DIR"),
|
|
||||||
"/../../crates-tauri/yaak-app/vendored/plugin-runtime/index.cjs"
|
|
||||||
));
|
|
||||||
static EMBEDDED_VENDORED_PLUGINS: Dir<'_> =
|
|
||||||
include_dir!("$CARGO_MANIFEST_DIR/../../crates-tauri/yaak-app/vendored/plugins");
|
|
||||||
|
|
||||||
pub struct CliContext {
|
|
||||||
data_dir: PathBuf,
|
|
||||||
query_manager: QueryManager,
|
|
||||||
blob_manager: BlobManager,
|
|
||||||
pub encryption_manager: Arc<EncryptionManager>,
|
|
||||||
plugin_manager: Option<Arc<PluginManager>>,
|
|
||||||
plugin_event_bridge: Mutex<Option<CliPluginEventBridge>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl CliContext {
|
|
||||||
pub async fn initialize(data_dir: PathBuf, app_id: &str, with_plugins: bool) -> Self {
|
|
||||||
let db_path = data_dir.join("db.sqlite");
|
|
||||||
let blob_path = data_dir.join("blobs.sqlite");
|
|
||||||
|
|
||||||
let (query_manager, blob_manager, _rx) = yaak_models::init_standalone(&db_path, &blob_path)
|
|
||||||
.expect("Failed to initialize database");
|
|
||||||
|
|
||||||
let encryption_manager = Arc::new(EncryptionManager::new(query_manager.clone(), app_id));
|
|
||||||
|
|
||||||
let plugin_manager = if with_plugins {
|
|
||||||
let embedded_vendored_plugin_dir = data_dir.join("vendored-plugins");
|
|
||||||
let bundled_plugin_dir =
|
|
||||||
resolve_bundled_plugin_dir_for_cli(&embedded_vendored_plugin_dir);
|
|
||||||
let installed_plugin_dir = data_dir.join("installed-plugins");
|
|
||||||
let node_bin_path = PathBuf::from("node");
|
|
||||||
|
|
||||||
if bundled_plugin_dir == embedded_vendored_plugin_dir {
|
|
||||||
prepare_embedded_vendored_plugins(&embedded_vendored_plugin_dir)
|
|
||||||
.expect("Failed to prepare bundled plugins");
|
|
||||||
}
|
|
||||||
|
|
||||||
let plugin_runtime_main =
|
|
||||||
std::env::var("YAAK_PLUGIN_RUNTIME").map(PathBuf::from).unwrap_or_else(|_| {
|
|
||||||
prepare_embedded_plugin_runtime(&data_dir)
|
|
||||||
.expect("Failed to prepare embedded plugin runtime")
|
|
||||||
});
|
|
||||||
|
|
||||||
match PluginManager::new(
|
|
||||||
bundled_plugin_dir,
|
|
||||||
embedded_vendored_plugin_dir,
|
|
||||||
installed_plugin_dir,
|
|
||||||
node_bin_path,
|
|
||||||
plugin_runtime_main,
|
|
||||||
&query_manager,
|
|
||||||
&PluginContext::new_empty(),
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
Ok(plugin_manager) => Some(Arc::new(plugin_manager)),
|
|
||||||
Err(err) => {
|
|
||||||
eprintln!("Warning: Failed to initialize plugins: {err}");
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
|
|
||||||
let plugin_event_bridge = if let Some(plugin_manager) = &plugin_manager {
|
|
||||||
Some(CliPluginEventBridge::start(plugin_manager.clone(), query_manager.clone()).await)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
|
|
||||||
Self {
|
|
||||||
data_dir,
|
|
||||||
query_manager,
|
|
||||||
blob_manager,
|
|
||||||
encryption_manager,
|
|
||||||
plugin_manager,
|
|
||||||
plugin_event_bridge: Mutex::new(plugin_event_bridge),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn data_dir(&self) -> &Path {
|
|
||||||
&self.data_dir
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn db(&self) -> DbContext<'_> {
|
|
||||||
self.query_manager.connect()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn query_manager(&self) -> &QueryManager {
|
|
||||||
&self.query_manager
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn blob_manager(&self) -> &BlobManager {
|
|
||||||
&self.blob_manager
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn plugin_manager(&self) -> Arc<PluginManager> {
|
|
||||||
self.plugin_manager.clone().expect("Plugin manager was not initialized for this command")
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn shutdown(&self) {
|
|
||||||
if let Some(plugin_manager) = &self.plugin_manager {
|
|
||||||
if let Some(plugin_event_bridge) = self.plugin_event_bridge.lock().await.take() {
|
|
||||||
plugin_event_bridge.shutdown(plugin_manager).await;
|
|
||||||
}
|
|
||||||
plugin_manager.terminate().await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn prepare_embedded_plugin_runtime(data_dir: &Path) -> std::io::Result<PathBuf> {
|
|
||||||
let runtime_dir = data_dir.join("vendored").join("plugin-runtime");
|
|
||||||
fs::create_dir_all(&runtime_dir)?;
|
|
||||||
let runtime_main = runtime_dir.join("index.cjs");
|
|
||||||
fs::write(&runtime_main, EMBEDDED_PLUGIN_RUNTIME)?;
|
|
||||||
Ok(runtime_main)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn prepare_embedded_vendored_plugins(vendored_plugin_dir: &Path) -> std::io::Result<()> {
|
|
||||||
fs::create_dir_all(vendored_plugin_dir)?;
|
|
||||||
EMBEDDED_VENDORED_PLUGINS.extract(vendored_plugin_dir)?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn resolve_bundled_plugin_dir_for_cli(embedded_vendored_plugin_dir: &Path) -> PathBuf {
|
|
||||||
if !cfg!(debug_assertions) {
|
|
||||||
return embedded_vendored_plugin_dir.to_path_buf();
|
|
||||||
}
|
|
||||||
|
|
||||||
let plugins_dir = match std::env::current_dir() {
|
|
||||||
Ok(cwd) => cwd.join("plugins"),
|
|
||||||
Err(_) => return embedded_vendored_plugin_dir.to_path_buf(),
|
|
||||||
};
|
|
||||||
|
|
||||||
if !plugins_dir.is_dir() {
|
|
||||||
return embedded_vendored_plugin_dir.to_path_buf();
|
|
||||||
}
|
|
||||||
|
|
||||||
plugins_dir.canonicalize().unwrap_or(plugins_dir)
|
|
||||||
}
|
|
||||||
@@ -1,101 +0,0 @@
|
|||||||
mod cli;
|
|
||||||
mod commands;
|
|
||||||
mod context;
|
|
||||||
mod plugin_events;
|
|
||||||
mod ui;
|
|
||||||
mod utils;
|
|
||||||
mod version;
|
|
||||||
|
|
||||||
use clap::Parser;
|
|
||||||
use cli::{Cli, Commands, RequestCommands};
|
|
||||||
use context::CliContext;
|
|
||||||
|
|
||||||
#[tokio::main]
|
|
||||||
async fn main() {
|
|
||||||
let Cli { data_dir, environment, verbose, log, command } = Cli::parse();
|
|
||||||
|
|
||||||
if let Some(log_level) = log {
|
|
||||||
match log_level {
|
|
||||||
Some(level) => {
|
|
||||||
env_logger::Builder::new().filter_level(level.as_filter()).init();
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
env_logger::Builder::from_env(env_logger::Env::default().default_filter_or("info"))
|
|
||||||
.init();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let app_id = if cfg!(debug_assertions) { "app.yaak.desktop.dev" } else { "app.yaak.desktop" };
|
|
||||||
|
|
||||||
let data_dir = data_dir.unwrap_or_else(|| {
|
|
||||||
dirs::data_dir().expect("Could not determine data directory").join(app_id)
|
|
||||||
});
|
|
||||||
|
|
||||||
let needs_context = matches!(
|
|
||||||
&command,
|
|
||||||
Commands::Send(_)
|
|
||||||
| Commands::Workspace(_)
|
|
||||||
| Commands::Request(_)
|
|
||||||
| Commands::Folder(_)
|
|
||||||
| Commands::Environment(_)
|
|
||||||
);
|
|
||||||
|
|
||||||
let needs_plugins = matches!(
|
|
||||||
&command,
|
|
||||||
Commands::Send(_)
|
|
||||||
| Commands::Request(cli::RequestArgs {
|
|
||||||
command: RequestCommands::Send { .. } | RequestCommands::Schema { .. },
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
let context = if needs_context {
|
|
||||||
Some(CliContext::initialize(data_dir, app_id, needs_plugins).await)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
|
|
||||||
let exit_code = match command {
|
|
||||||
Commands::Auth(args) => commands::auth::run(args).await,
|
|
||||||
Commands::Plugin(args) => commands::plugin::run(args).await,
|
|
||||||
Commands::Build(args) => commands::plugin::run_build(args).await,
|
|
||||||
Commands::Dev(args) => commands::plugin::run_dev(args).await,
|
|
||||||
Commands::Send(args) => {
|
|
||||||
commands::send::run(
|
|
||||||
context.as_ref().expect("context initialized for send"),
|
|
||||||
args,
|
|
||||||
environment.as_deref(),
|
|
||||||
verbose,
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
Commands::Workspace(args) => commands::workspace::run(
|
|
||||||
context.as_ref().expect("context initialized for workspace"),
|
|
||||||
args,
|
|
||||||
),
|
|
||||||
Commands::Request(args) => {
|
|
||||||
commands::request::run(
|
|
||||||
context.as_ref().expect("context initialized for request"),
|
|
||||||
args,
|
|
||||||
environment.as_deref(),
|
|
||||||
verbose,
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
Commands::Folder(args) => {
|
|
||||||
commands::folder::run(context.as_ref().expect("context initialized for folder"), args)
|
|
||||||
}
|
|
||||||
Commands::Environment(args) => commands::environment::run(
|
|
||||||
context.as_ref().expect("context initialized for environment"),
|
|
||||||
args,
|
|
||||||
),
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(context) = &context {
|
|
||||||
context.shutdown().await;
|
|
||||||
}
|
|
||||||
|
|
||||||
if exit_code != 0 {
|
|
||||||
std::process::exit(exit_code);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,212 +0,0 @@
|
|||||||
use std::sync::Arc;
|
|
||||||
use tokio::task::JoinHandle;
|
|
||||||
use yaak::plugin_events::{
|
|
||||||
GroupedPluginEvent, HostRequest, SharedPluginEventContext, handle_shared_plugin_event,
|
|
||||||
};
|
|
||||||
use yaak_models::query_manager::QueryManager;
|
|
||||||
use yaak_plugins::events::{
|
|
||||||
EmptyPayload, ErrorResponse, InternalEvent, InternalEventPayload, ListOpenWorkspacesResponse,
|
|
||||||
WorkspaceInfo,
|
|
||||||
};
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
|
|
||||||
pub struct CliPluginEventBridge {
|
|
||||||
rx_id: String,
|
|
||||||
task: JoinHandle<()>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl CliPluginEventBridge {
|
|
||||||
pub async fn start(plugin_manager: Arc<PluginManager>, query_manager: QueryManager) -> Self {
|
|
||||||
let (rx_id, mut rx) = plugin_manager.subscribe("cli").await;
|
|
||||||
let rx_id_for_task = rx_id.clone();
|
|
||||||
let pm = plugin_manager.clone();
|
|
||||||
|
|
||||||
let task = tokio::spawn(async move {
|
|
||||||
while let Some(event) = rx.recv().await {
|
|
||||||
// Events with reply IDs are replies to app-originated requests.
|
|
||||||
if event.reply_id.is_some() {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let Some(plugin_handle) = pm.get_plugin_by_ref_id(&event.plugin_ref_id).await
|
|
||||||
else {
|
|
||||||
eprintln!(
|
|
||||||
"Warning: Ignoring plugin event with unknown plugin ref '{}'",
|
|
||||||
event.plugin_ref_id
|
|
||||||
);
|
|
||||||
continue;
|
|
||||||
};
|
|
||||||
|
|
||||||
let plugin_name = plugin_handle.info().name;
|
|
||||||
let Some(reply_payload) = build_plugin_reply(&query_manager, &event, &plugin_name)
|
|
||||||
else {
|
|
||||||
continue;
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Err(err) = pm.reply(&event, &reply_payload).await {
|
|
||||||
eprintln!("Warning: Failed replying to plugin event: {err}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pm.unsubscribe(&rx_id_for_task).await;
|
|
||||||
});
|
|
||||||
|
|
||||||
Self { rx_id, task }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn shutdown(self, plugin_manager: &PluginManager) {
|
|
||||||
plugin_manager.unsubscribe(&self.rx_id).await;
|
|
||||||
self.task.abort();
|
|
||||||
let _ = self.task.await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn build_plugin_reply(
|
|
||||||
query_manager: &QueryManager,
|
|
||||||
event: &InternalEvent,
|
|
||||||
plugin_name: &str,
|
|
||||||
) -> Option<InternalEventPayload> {
|
|
||||||
match handle_shared_plugin_event(
|
|
||||||
query_manager,
|
|
||||||
&event.payload,
|
|
||||||
SharedPluginEventContext {
|
|
||||||
plugin_name,
|
|
||||||
workspace_id: event.context.workspace_id.as_deref(),
|
|
||||||
},
|
|
||||||
) {
|
|
||||||
GroupedPluginEvent::Handled(payload) => payload,
|
|
||||||
GroupedPluginEvent::ToHandle(host_request) => match host_request {
|
|
||||||
HostRequest::ErrorResponse(resp) => {
|
|
||||||
eprintln!("[plugin:{}] error: {}", plugin_name, resp.error);
|
|
||||||
None
|
|
||||||
}
|
|
||||||
HostRequest::ReloadResponse(_) => None,
|
|
||||||
HostRequest::ShowToast(req) => {
|
|
||||||
eprintln!("[plugin:{}] {}", plugin_name, req.message);
|
|
||||||
Some(InternalEventPayload::ShowToastResponse(EmptyPayload {}))
|
|
||||||
}
|
|
||||||
HostRequest::ListOpenWorkspaces(_) => {
|
|
||||||
let workspaces = match query_manager.connect().list_workspaces() {
|
|
||||||
Ok(workspaces) => workspaces
|
|
||||||
.into_iter()
|
|
||||||
.map(|w| WorkspaceInfo { id: w.id.clone(), name: w.name, label: w.id })
|
|
||||||
.collect(),
|
|
||||||
Err(err) => {
|
|
||||||
return Some(InternalEventPayload::ErrorResponse(ErrorResponse {
|
|
||||||
error: format!("Failed to list workspaces in CLI: {err}"),
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
Some(InternalEventPayload::ListOpenWorkspacesResponse(ListOpenWorkspacesResponse {
|
|
||||||
workspaces,
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
req => Some(InternalEventPayload::ErrorResponse(ErrorResponse {
|
|
||||||
error: format!("Unsupported plugin request in CLI: {}", req.type_name()),
|
|
||||||
})),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod tests {
|
|
||||||
use super::*;
|
|
||||||
use tempfile::TempDir;
|
|
||||||
use yaak_plugins::events::{GetKeyValueRequest, PluginContext, WindowInfoRequest};
|
|
||||||
|
|
||||||
fn query_manager_for_test() -> (QueryManager, TempDir) {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let db_path = temp_dir.path().join("db.sqlite");
|
|
||||||
let blob_path = temp_dir.path().join("blobs.sqlite");
|
|
||||||
let (query_manager, _blob_manager, _rx) =
|
|
||||||
yaak_models::init_standalone(&db_path, &blob_path).expect("Failed to initialize DB");
|
|
||||||
(query_manager, temp_dir)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn event(payload: InternalEventPayload) -> InternalEvent {
|
|
||||||
InternalEvent {
|
|
||||||
id: "evt_1".to_string(),
|
|
||||||
plugin_ref_id: "plugin_ref_1".to_string(),
|
|
||||||
plugin_name: "@yaak/test-plugin".to_string(),
|
|
||||||
reply_id: None,
|
|
||||||
context: PluginContext::new_empty(),
|
|
||||||
payload,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn key_value_requests_round_trip() {
|
|
||||||
let (query_manager, _temp_dir) = query_manager_for_test();
|
|
||||||
let plugin_name = "@yaak/test-plugin";
|
|
||||||
|
|
||||||
let get_missing = build_plugin_reply(
|
|
||||||
&query_manager,
|
|
||||||
&event(InternalEventPayload::GetKeyValueRequest(GetKeyValueRequest {
|
|
||||||
key: "missing".to_string(),
|
|
||||||
})),
|
|
||||||
plugin_name,
|
|
||||||
);
|
|
||||||
match get_missing {
|
|
||||||
Some(InternalEventPayload::GetKeyValueResponse(r)) => assert_eq!(r.value, None),
|
|
||||||
other => panic!("unexpected payload for missing get: {other:?}"),
|
|
||||||
}
|
|
||||||
|
|
||||||
let set = build_plugin_reply(
|
|
||||||
&query_manager,
|
|
||||||
&event(InternalEventPayload::SetKeyValueRequest(
|
|
||||||
yaak_plugins::events::SetKeyValueRequest {
|
|
||||||
key: "token".to_string(),
|
|
||||||
value: "{\"access_token\":\"abc\"}".to_string(),
|
|
||||||
},
|
|
||||||
)),
|
|
||||||
plugin_name,
|
|
||||||
);
|
|
||||||
assert!(matches!(set, Some(InternalEventPayload::SetKeyValueResponse(_))));
|
|
||||||
|
|
||||||
let get_present = build_plugin_reply(
|
|
||||||
&query_manager,
|
|
||||||
&event(InternalEventPayload::GetKeyValueRequest(GetKeyValueRequest {
|
|
||||||
key: "token".to_string(),
|
|
||||||
})),
|
|
||||||
plugin_name,
|
|
||||||
);
|
|
||||||
match get_present {
|
|
||||||
Some(InternalEventPayload::GetKeyValueResponse(r)) => {
|
|
||||||
assert_eq!(r.value, Some("{\"access_token\":\"abc\"}".to_string()))
|
|
||||||
}
|
|
||||||
other => panic!("unexpected payload for present get: {other:?}"),
|
|
||||||
}
|
|
||||||
|
|
||||||
let delete = build_plugin_reply(
|
|
||||||
&query_manager,
|
|
||||||
&event(InternalEventPayload::DeleteKeyValueRequest(
|
|
||||||
yaak_plugins::events::DeleteKeyValueRequest { key: "token".to_string() },
|
|
||||||
)),
|
|
||||||
plugin_name,
|
|
||||||
);
|
|
||||||
match delete {
|
|
||||||
Some(InternalEventPayload::DeleteKeyValueResponse(r)) => assert!(r.deleted),
|
|
||||||
other => panic!("unexpected payload for delete: {other:?}"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn unsupported_request_gets_error_reply() {
|
|
||||||
let (query_manager, _temp_dir) = query_manager_for_test();
|
|
||||||
let payload = build_plugin_reply(
|
|
||||||
&query_manager,
|
|
||||||
&event(InternalEventPayload::WindowInfoRequest(WindowInfoRequest {
|
|
||||||
label: "main".to_string(),
|
|
||||||
})),
|
|
||||||
"@yaak/test-plugin",
|
|
||||||
);
|
|
||||||
|
|
||||||
match payload {
|
|
||||||
Some(InternalEventPayload::ErrorResponse(err)) => {
|
|
||||||
assert!(err.error.contains("Unsupported plugin request in CLI"));
|
|
||||||
assert!(err.error.contains("window_info_request"));
|
|
||||||
}
|
|
||||||
other => panic!("unexpected payload for unsupported request: {other:?}"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
use console::style;
|
|
||||||
use std::io::{self, IsTerminal};
|
|
||||||
|
|
||||||
pub fn info(message: &str) {
|
|
||||||
if io::stdout().is_terminal() {
|
|
||||||
println!("{:<8} {}", style("INFO").cyan().bold(), style(message).cyan());
|
|
||||||
} else {
|
|
||||||
println!("INFO {message}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn warning(message: &str) {
|
|
||||||
if io::stdout().is_terminal() {
|
|
||||||
println!("{:<8} {}", style("WARNING").yellow().bold(), style(message).yellow());
|
|
||||||
} else {
|
|
||||||
println!("WARNING {message}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn success(message: &str) {
|
|
||||||
if io::stdout().is_terminal() {
|
|
||||||
println!("{:<8} {}", style("SUCCESS").green().bold(), style(message).green());
|
|
||||||
} else {
|
|
||||||
println!("SUCCESS {message}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn error(message: &str) {
|
|
||||||
if io::stderr().is_terminal() {
|
|
||||||
eprintln!("{:<8} {}", style("ERROR").red().bold(), style(message).red());
|
|
||||||
} else {
|
|
||||||
eprintln!("Error: {message}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
use std::io::{self, IsTerminal, Write};
|
|
||||||
|
|
||||||
pub fn confirm_delete(resource_name: &str, resource_id: &str) -> bool {
|
|
||||||
if !io::stdin().is_terminal() {
|
|
||||||
eprintln!("Refusing to delete in non-interactive mode without --yes");
|
|
||||||
std::process::exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
print!("Delete {resource_name} {resource_id}? [y/N]: ");
|
|
||||||
io::stdout().flush().expect("Failed to flush stdout");
|
|
||||||
|
|
||||||
let mut input = String::new();
|
|
||||||
io::stdin().read_line(&mut input).expect("Failed to read confirmation");
|
|
||||||
|
|
||||||
matches!(input.trim().to_lowercase().as_str(), "y" | "yes")
|
|
||||||
}
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
use reqwest::Client;
|
|
||||||
use reqwest::header::{HeaderMap, HeaderName, HeaderValue, USER_AGENT};
|
|
||||||
use serde_json::Value;
|
|
||||||
|
|
||||||
pub fn build_client(session_token: Option<&str>) -> Result<Client, String> {
|
|
||||||
let mut headers = HeaderMap::new();
|
|
||||||
let user_agent = HeaderValue::from_str(&user_agent())
|
|
||||||
.map_err(|e| format!("Failed to build user-agent header: {e}"))?;
|
|
||||||
headers.insert(USER_AGENT, user_agent);
|
|
||||||
|
|
||||||
if let Some(token) = session_token {
|
|
||||||
let token_value = HeaderValue::from_str(token)
|
|
||||||
.map_err(|e| format!("Failed to build session header: {e}"))?;
|
|
||||||
headers.insert(HeaderName::from_static("x-yaak-session"), token_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
Client::builder()
|
|
||||||
.default_headers(headers)
|
|
||||||
.build()
|
|
||||||
.map_err(|e| format!("Failed to initialize HTTP client: {e}"))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_api_error(status: u16, body: &str) -> String {
|
|
||||||
if let Ok(value) = serde_json::from_str::<Value>(body) {
|
|
||||||
if let Some(message) = value.get("message").and_then(Value::as_str) {
|
|
||||||
return message.to_string();
|
|
||||||
}
|
|
||||||
if let Some(error) = value.get("error").and_then(Value::as_str) {
|
|
||||||
return error.to_string();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
format!("API error {status}: {body}")
|
|
||||||
}
|
|
||||||
|
|
||||||
fn user_agent() -> String {
|
|
||||||
format!("YaakCli/{} ({})", crate::version::cli_version(), ua_platform())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn ua_platform() -> &'static str {
|
|
||||||
match std::env::consts::OS {
|
|
||||||
"windows" => "Win",
|
|
||||||
"darwin" => "Mac",
|
|
||||||
"linux" => "Linux",
|
|
||||||
_ => "Unknown",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,131 +0,0 @@
|
|||||||
use serde::Serialize;
|
|
||||||
use serde::de::DeserializeOwned;
|
|
||||||
use serde_json::{Map, Value};
|
|
||||||
|
|
||||||
type JsonResult<T> = std::result::Result<T, String>;
|
|
||||||
|
|
||||||
pub fn is_json_shorthand(input: &str) -> bool {
|
|
||||||
input.trim_start().starts_with('{')
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_json_object(raw: &str, context: &str) -> JsonResult<Value> {
|
|
||||||
let value: Value = serde_json::from_str(raw)
|
|
||||||
.map_err(|error| format!("Invalid JSON for {context}: {error}"))?;
|
|
||||||
|
|
||||||
if !value.is_object() {
|
|
||||||
return Err(format!("JSON payload for {context} must be an object"));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_optional_json(
|
|
||||||
json_flag: Option<String>,
|
|
||||||
json_shorthand: Option<String>,
|
|
||||||
context: &str,
|
|
||||||
) -> JsonResult<Option<Value>> {
|
|
||||||
match (json_flag, json_shorthand) {
|
|
||||||
(Some(_), Some(_)) => {
|
|
||||||
Err(format!("Cannot provide both --json and positional JSON for {context}"))
|
|
||||||
}
|
|
||||||
(Some(raw), None) => parse_json_object(&raw, context).map(Some),
|
|
||||||
(None, Some(raw)) => parse_json_object(&raw, context).map(Some),
|
|
||||||
(None, None) => Ok(None),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_required_json(
|
|
||||||
json_flag: Option<String>,
|
|
||||||
json_shorthand: Option<String>,
|
|
||||||
context: &str,
|
|
||||||
) -> JsonResult<Value> {
|
|
||||||
parse_optional_json(json_flag, json_shorthand, context)?
|
|
||||||
.ok_or_else(|| format!("Missing JSON payload for {context}. Use --json or positional JSON"))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn require_id(payload: &Value, context: &str) -> JsonResult<String> {
|
|
||||||
payload
|
|
||||||
.get("id")
|
|
||||||
.and_then(|value| value.as_str())
|
|
||||||
.filter(|value| !value.is_empty())
|
|
||||||
.map(|value| value.to_string())
|
|
||||||
.ok_or_else(|| format!("{context} requires a non-empty \"id\" field"))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn validate_create_id(payload: &Value, context: &str) -> JsonResult<()> {
|
|
||||||
let Some(id_value) = payload.get("id") else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
match id_value {
|
|
||||||
Value::String(id) if id.is_empty() => Ok(()),
|
|
||||||
_ => Err(format!("{context} create JSON must omit \"id\" or set it to an empty string")),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn merge_workspace_id_arg(
|
|
||||||
workspace_id_from_arg: Option<&str>,
|
|
||||||
payload_workspace_id: &mut String,
|
|
||||||
context: &str,
|
|
||||||
) -> JsonResult<()> {
|
|
||||||
if let Some(workspace_id_arg) = workspace_id_from_arg {
|
|
||||||
if payload_workspace_id.is_empty() {
|
|
||||||
*payload_workspace_id = workspace_id_arg.to_string();
|
|
||||||
} else if payload_workspace_id != workspace_id_arg {
|
|
||||||
return Err(format!(
|
|
||||||
"{context} got conflicting workspace_id values between positional arg and JSON payload"
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if payload_workspace_id.is_empty() {
|
|
||||||
return Err(format!(
|
|
||||||
"{context} requires non-empty \"workspaceId\" in JSON payload or positional workspace_id"
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn apply_merge_patch<T>(existing: &T, patch: &Value, id: &str, context: &str) -> JsonResult<T>
|
|
||||||
where
|
|
||||||
T: Serialize + DeserializeOwned,
|
|
||||||
{
|
|
||||||
let mut base = serde_json::to_value(existing)
|
|
||||||
.map_err(|error| format!("Failed to serialize existing model for {context}: {error}"))?;
|
|
||||||
merge_patch(&mut base, patch);
|
|
||||||
|
|
||||||
let Some(base_object) = base.as_object_mut() else {
|
|
||||||
return Err(format!("Merged payload for {context} must be an object"));
|
|
||||||
};
|
|
||||||
base_object.insert("id".to_string(), Value::String(id.to_string()));
|
|
||||||
|
|
||||||
serde_json::from_value(base)
|
|
||||||
.map_err(|error| format!("Failed to deserialize merged payload for {context}: {error}"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn merge_patch(target: &mut Value, patch: &Value) {
|
|
||||||
match patch {
|
|
||||||
Value::Object(patch_map) => {
|
|
||||||
if !target.is_object() {
|
|
||||||
*target = Value::Object(Map::new());
|
|
||||||
}
|
|
||||||
|
|
||||||
let target_map =
|
|
||||||
target.as_object_mut().expect("merge_patch target expected to be object");
|
|
||||||
|
|
||||||
for (key, patch_value) in patch_map {
|
|
||||||
if patch_value.is_null() {
|
|
||||||
target_map.remove(key);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let target_entry = target_map.entry(key.clone()).or_insert(Value::Null);
|
|
||||||
merge_patch(target_entry, patch_value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
*target = patch.clone();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
pub mod confirm;
|
|
||||||
pub mod http;
|
|
||||||
pub mod json;
|
|
||||||
pub mod schema;
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
use serde_json::{Value, json};
|
|
||||||
|
|
||||||
pub fn append_agent_hints(schema: &mut Value) {
|
|
||||||
let Some(schema_obj) = schema.as_object_mut() else {
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
|
|
||||||
schema_obj.insert(
|
|
||||||
"x-yaak-agent-hints".to_string(),
|
|
||||||
json!({
|
|
||||||
"templateVariableSyntax": "${[ my_var ]}",
|
|
||||||
"templateFunctionSyntax": "${[ namespace.my_func(a='aaa',b='bbb') ]}",
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
pub fn cli_version() -> &'static str {
|
|
||||||
option_env!("YAAK_CLI_VERSION").unwrap_or(env!("CARGO_PKG_VERSION"))
|
|
||||||
}
|
|
||||||
@@ -1,42 +0,0 @@
|
|||||||
use std::io::{Read, Write};
|
|
||||||
use std::net::TcpListener;
|
|
||||||
use std::thread;
|
|
||||||
|
|
||||||
pub struct TestHttpServer {
|
|
||||||
pub url: String,
|
|
||||||
handle: Option<thread::JoinHandle<()>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TestHttpServer {
|
|
||||||
pub fn spawn_ok(body: &'static str) -> Self {
|
|
||||||
let listener = TcpListener::bind("127.0.0.1:0").expect("Failed to bind test HTTP server");
|
|
||||||
let addr = listener.local_addr().expect("Failed to get local addr");
|
|
||||||
let url = format!("http://{addr}/test");
|
|
||||||
let body_bytes = body.as_bytes().to_vec();
|
|
||||||
|
|
||||||
let handle = thread::spawn(move || {
|
|
||||||
if let Ok((mut stream, _)) = listener.accept() {
|
|
||||||
let mut request_buf = [0u8; 4096];
|
|
||||||
let _ = stream.read(&mut request_buf);
|
|
||||||
|
|
||||||
let response = format!(
|
|
||||||
"HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\nContent-Length: {}\r\nConnection: close\r\n\r\n",
|
|
||||||
body_bytes.len()
|
|
||||||
);
|
|
||||||
let _ = stream.write_all(response.as_bytes());
|
|
||||||
let _ = stream.write_all(&body_bytes);
|
|
||||||
let _ = stream.flush();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
Self { url, handle: Some(handle) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for TestHttpServer {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
if let Some(handle) = self.handle.take() {
|
|
||||||
let _ = handle.join();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
#![allow(dead_code)]
|
|
||||||
|
|
||||||
pub mod http_server;
|
|
||||||
|
|
||||||
use assert_cmd::Command;
|
|
||||||
use assert_cmd::cargo::cargo_bin_cmd;
|
|
||||||
use std::path::Path;
|
|
||||||
use yaak_models::models::{Folder, GrpcRequest, HttpRequest, WebsocketRequest, Workspace};
|
|
||||||
use yaak_models::query_manager::QueryManager;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
pub fn cli_cmd(data_dir: &Path) -> Command {
|
|
||||||
let mut cmd = cargo_bin_cmd!("yaak");
|
|
||||||
cmd.arg("--data-dir").arg(data_dir);
|
|
||||||
cmd
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_created_id(stdout: &[u8], label: &str) -> String {
|
|
||||||
String::from_utf8_lossy(stdout)
|
|
||||||
.trim()
|
|
||||||
.split_once(": ")
|
|
||||||
.map(|(_, id)| id.to_string())
|
|
||||||
.unwrap_or_else(|| panic!("Expected id in '{label}' output"))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn query_manager(data_dir: &Path) -> QueryManager {
|
|
||||||
let db_path = data_dir.join("db.sqlite");
|
|
||||||
let blob_path = data_dir.join("blobs.sqlite");
|
|
||||||
let (query_manager, _blob_manager, _rx) =
|
|
||||||
yaak_models::init_standalone(&db_path, &blob_path).expect("Failed to initialize DB");
|
|
||||||
query_manager
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn seed_workspace(data_dir: &Path, workspace_id: &str) {
|
|
||||||
let workspace = Workspace {
|
|
||||||
id: workspace_id.to_string(),
|
|
||||||
name: "Seed Workspace".to_string(),
|
|
||||||
description: "Seeded for integration tests".to_string(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.upsert_workspace(&workspace, &UpdateSource::Sync)
|
|
||||||
.expect("Failed to seed workspace");
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn seed_request(data_dir: &Path, workspace_id: &str, request_id: &str) {
|
|
||||||
let request = HttpRequest {
|
|
||||||
id: request_id.to_string(),
|
|
||||||
workspace_id: workspace_id.to_string(),
|
|
||||||
name: "Seeded Request".to_string(),
|
|
||||||
method: "GET".to_string(),
|
|
||||||
url: "https://example.com".to_string(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.upsert_http_request(&request, &UpdateSource::Sync)
|
|
||||||
.expect("Failed to seed request");
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn seed_folder(data_dir: &Path, workspace_id: &str, folder_id: &str) {
|
|
||||||
let folder = Folder {
|
|
||||||
id: folder_id.to_string(),
|
|
||||||
workspace_id: workspace_id.to_string(),
|
|
||||||
name: "Seed Folder".to_string(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.upsert_folder(&folder, &UpdateSource::Sync)
|
|
||||||
.expect("Failed to seed folder");
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn seed_grpc_request(data_dir: &Path, workspace_id: &str, request_id: &str) {
|
|
||||||
let request = GrpcRequest {
|
|
||||||
id: request_id.to_string(),
|
|
||||||
workspace_id: workspace_id.to_string(),
|
|
||||||
name: "Seeded gRPC Request".to_string(),
|
|
||||||
url: "https://example.com".to_string(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.upsert_grpc_request(&request, &UpdateSource::Sync)
|
|
||||||
.expect("Failed to seed gRPC request");
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn seed_websocket_request(data_dir: &Path, workspace_id: &str, request_id: &str) {
|
|
||||||
let request = WebsocketRequest {
|
|
||||||
id: request_id.to_string(),
|
|
||||||
workspace_id: workspace_id.to_string(),
|
|
||||||
name: "Seeded WebSocket Request".to_string(),
|
|
||||||
url: "wss://example.com/socket".to_string(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.upsert_websocket_request(&request, &UpdateSource::Sync)
|
|
||||||
.expect("Failed to seed WebSocket request");
|
|
||||||
}
|
|
||||||
@@ -1,146 +0,0 @@
|
|||||||
mod common;
|
|
||||||
|
|
||||||
use common::{cli_cmd, parse_created_id, query_manager, seed_workspace};
|
|
||||||
use predicates::str::contains;
|
|
||||||
use tempfile::TempDir;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_list_show_delete_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["environment", "list", "wk_test"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("Global Variables"));
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args(["environment", "create", "wk_test", "--name", "Production"])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let environment_id = parse_created_id(&create_assert.get_output().stdout, "environment create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["environment", "list", "wk_test"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(&environment_id))
|
|
||||||
.stdout(contains("Production"));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["environment", "show", &environment_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("\"id\": \"{environment_id}\"")))
|
|
||||||
.stdout(contains("\"parentModel\": \"environment\""));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["environment", "delete", &environment_id, "--yes"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Deleted environment: {environment_id}")));
|
|
||||||
|
|
||||||
assert!(query_manager(data_dir).connect().get_environment(&environment_id).is_err());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn json_create_and_update_merge_patch_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"environment",
|
|
||||||
"create",
|
|
||||||
r#"{"workspaceId":"wk_test","name":"Json Environment"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let environment_id = parse_created_id(&create_assert.get_output().stdout, "environment create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"environment",
|
|
||||||
"update",
|
|
||||||
&format!(r##"{{"id":"{}","color":"#00ff00"}}"##, environment_id),
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Updated environment: {environment_id}")));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["environment", "show", &environment_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"name\": \"Json Environment\""))
|
|
||||||
.stdout(contains("\"color\": \"#00ff00\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_merges_positional_workspace_id_into_json_payload() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"environment",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--json",
|
|
||||||
r#"{"name":"Merged Environment"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let environment_id = parse_created_id(&create_assert.get_output().stdout, "environment create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["environment", "show", &environment_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"workspaceId\": \"wk_test\""))
|
|
||||||
.stdout(contains("\"name\": \"Merged Environment\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_rejects_conflicting_workspace_ids_between_arg_and_json() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
seed_workspace(data_dir, "wk_other");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"environment",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--json",
|
|
||||||
r#"{"workspaceId":"wk_other","name":"Mismatch"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.stderr(contains(
|
|
||||||
"environment create got conflicting workspace_id values between positional arg and JSON payload",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn environment_schema_outputs_json_schema() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["environment", "schema"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"type\":\"object\""))
|
|
||||||
.stdout(contains("\"x-yaak-agent-hints\""))
|
|
||||||
.stdout(contains("\"templateVariableSyntax\":\"${[ my_var ]}\""))
|
|
||||||
.stdout(contains(
|
|
||||||
"\"templateFunctionSyntax\":\"${[ namespace.my_func(a='aaa',b='bbb') ]}\"",
|
|
||||||
))
|
|
||||||
.stdout(contains("\"workspaceId\""));
|
|
||||||
}
|
|
||||||
@@ -1,122 +0,0 @@
|
|||||||
mod common;
|
|
||||||
|
|
||||||
use common::{cli_cmd, parse_created_id, query_manager, seed_workspace};
|
|
||||||
use predicates::str::contains;
|
|
||||||
use tempfile::TempDir;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_list_show_delete_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args(["folder", "create", "wk_test", "--name", "Auth"])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let folder_id = parse_created_id(&create_assert.get_output().stdout, "folder create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["folder", "list", "wk_test"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(&folder_id))
|
|
||||||
.stdout(contains("Auth"));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["folder", "show", &folder_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("\"id\": \"{folder_id}\"")))
|
|
||||||
.stdout(contains("\"workspaceId\": \"wk_test\""));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["folder", "delete", &folder_id, "--yes"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Deleted folder: {folder_id}")));
|
|
||||||
|
|
||||||
assert!(query_manager(data_dir).connect().get_folder(&folder_id).is_err());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn json_create_and_update_merge_patch_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"folder",
|
|
||||||
"create",
|
|
||||||
r#"{"workspaceId":"wk_test","name":"Json Folder"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let folder_id = parse_created_id(&create_assert.get_output().stdout, "folder create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"folder",
|
|
||||||
"update",
|
|
||||||
&format!(r#"{{"id":"{}","description":"Folder Description"}}"#, folder_id),
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Updated folder: {folder_id}")));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["folder", "show", &folder_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"name\": \"Json Folder\""))
|
|
||||||
.stdout(contains("\"description\": \"Folder Description\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_merges_positional_workspace_id_into_json_payload() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"folder",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--json",
|
|
||||||
r#"{"name":"Merged Folder"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let folder_id = parse_created_id(&create_assert.get_output().stdout, "folder create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["folder", "show", &folder_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"workspaceId\": \"wk_test\""))
|
|
||||||
.stdout(contains("\"name\": \"Merged Folder\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_rejects_conflicting_workspace_ids_between_arg_and_json() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
seed_workspace(data_dir, "wk_other");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"folder",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--json",
|
|
||||||
r#"{"workspaceId":"wk_other","name":"Mismatch"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.stderr(contains(
|
|
||||||
"folder create got conflicting workspace_id values between positional arg and JSON payload",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
@@ -1,291 +0,0 @@
|
|||||||
mod common;
|
|
||||||
|
|
||||||
use common::http_server::TestHttpServer;
|
|
||||||
use common::{
|
|
||||||
cli_cmd, parse_created_id, query_manager, seed_grpc_request, seed_request,
|
|
||||||
seed_websocket_request, seed_workspace,
|
|
||||||
};
|
|
||||||
use predicates::str::contains;
|
|
||||||
use tempfile::TempDir;
|
|
||||||
use yaak_models::models::HttpResponseState;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn show_and_delete_yes_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"request",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--name",
|
|
||||||
"Smoke Test",
|
|
||||||
"--url",
|
|
||||||
"https://example.com",
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
|
|
||||||
let request_id = parse_created_id(&create_assert.get_output().stdout, "request create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "show", &request_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("\"id\": \"{request_id}\"")))
|
|
||||||
.stdout(contains("\"workspaceId\": \"wk_test\""));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "delete", &request_id, "--yes"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Deleted request: {request_id}")));
|
|
||||||
|
|
||||||
assert!(query_manager(data_dir).connect().get_http_request(&request_id).is_err());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn delete_without_yes_fails_in_non_interactive_mode() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
seed_request(data_dir, "wk_test", "rq_seed_delete_noninteractive");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "delete", "rq_seed_delete_noninteractive"])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.code(1)
|
|
||||||
.stderr(contains("Refusing to delete in non-interactive mode without --yes"));
|
|
||||||
|
|
||||||
assert!(
|
|
||||||
query_manager(data_dir).connect().get_http_request("rq_seed_delete_noninteractive").is_ok()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn json_create_and_update_merge_patch_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"request",
|
|
||||||
"create",
|
|
||||||
r#"{"workspaceId":"wk_test","name":"Json Request","url":"https://example.com"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let request_id = parse_created_id(&create_assert.get_output().stdout, "request create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"request",
|
|
||||||
"update",
|
|
||||||
&format!(r#"{{"id":"{}","name":"Renamed Request"}}"#, request_id),
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Updated request: {request_id}")));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "show", &request_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"name\": \"Renamed Request\""))
|
|
||||||
.stdout(contains("\"url\": \"https://example.com\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn update_requires_id_in_json_payload() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "update", r#"{"name":"No ID"}"#])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.stderr(contains("request update requires a non-empty \"id\" field"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_allows_workspace_only_with_empty_defaults() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir).args(["request", "create", "wk_test"]).assert().success();
|
|
||||||
let request_id = parse_created_id(&create_assert.get_output().stdout, "request create");
|
|
||||||
|
|
||||||
let request = query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.get_http_request(&request_id)
|
|
||||||
.expect("Failed to load created request");
|
|
||||||
assert_eq!(request.workspace_id, "wk_test");
|
|
||||||
assert_eq!(request.method, "GET");
|
|
||||||
assert_eq!(request.name, "");
|
|
||||||
assert_eq!(request.url, "");
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_merges_positional_workspace_id_into_json_payload() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"request",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--json",
|
|
||||||
r#"{"name":"Merged Request","url":"https://example.com"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let request_id = parse_created_id(&create_assert.get_output().stdout, "request create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "show", &request_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"workspaceId\": \"wk_test\""))
|
|
||||||
.stdout(contains("\"name\": \"Merged Request\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_rejects_conflicting_workspace_ids_between_arg_and_json() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
seed_workspace(data_dir, "wk_other");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"request",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--json",
|
|
||||||
r#"{"workspaceId":"wk_other","name":"Mismatch"}"#,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.stderr(contains(
|
|
||||||
"request create got conflicting workspace_id values between positional arg and JSON payload",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn request_send_persists_response_body_and_events() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let server = TestHttpServer::spawn_ok("hello from integration test");
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"request",
|
|
||||||
"create",
|
|
||||||
"wk_test",
|
|
||||||
"--name",
|
|
||||||
"Send Test",
|
|
||||||
"--url",
|
|
||||||
&server.url,
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let request_id = parse_created_id(&create_assert.get_output().stdout, "request create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "send", &request_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("hello from integration test"));
|
|
||||||
|
|
||||||
let qm = query_manager(data_dir);
|
|
||||||
let db = qm.connect();
|
|
||||||
let responses =
|
|
||||||
db.list_http_responses_for_request(&request_id, None).expect("Failed to load responses");
|
|
||||||
assert_eq!(responses.len(), 1, "expected exactly one persisted response");
|
|
||||||
|
|
||||||
let response = &responses[0];
|
|
||||||
assert_eq!(response.status, 200);
|
|
||||||
assert!(matches!(response.state, HttpResponseState::Closed));
|
|
||||||
assert!(response.error.is_none());
|
|
||||||
|
|
||||||
let body_path =
|
|
||||||
response.body_path.as_ref().expect("expected persisted response body path").to_string();
|
|
||||||
let body = std::fs::read_to_string(&body_path).expect("Failed to read response body file");
|
|
||||||
assert_eq!(body, "hello from integration test");
|
|
||||||
|
|
||||||
let events =
|
|
||||||
db.list_http_response_events(&response.id).expect("Failed to load response events");
|
|
||||||
assert!(!events.is_empty(), "expected at least one persisted response event");
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn request_schema_http_outputs_json_schema() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "schema", "http"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"type\":\"object\""))
|
|
||||||
.stdout(contains("\"x-yaak-agent-hints\""))
|
|
||||||
.stdout(contains("\"templateVariableSyntax\":\"${[ my_var ]}\""))
|
|
||||||
.stdout(contains(
|
|
||||||
"\"templateFunctionSyntax\":\"${[ namespace.my_func(a='aaa',b='bbb') ]}\"",
|
|
||||||
))
|
|
||||||
.stdout(contains("\"authentication\":"))
|
|
||||||
.stdout(contains("/foo/:id/comments/:commentId"))
|
|
||||||
.stdout(contains("put concrete values in `urlParameters`"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn request_schema_http_pretty_prints_with_flag() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "schema", "http", "--pretty"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"type\": \"object\""))
|
|
||||||
.stdout(contains("\"authentication\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn request_send_grpc_returns_explicit_nyi_error() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
seed_grpc_request(data_dir, "wk_test", "gr_seed_nyi");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "send", "gr_seed_nyi"])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.code(1)
|
|
||||||
.stderr(contains("gRPC request send is not implemented yet in yaak-cli"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn request_send_websocket_returns_explicit_nyi_error() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
seed_websocket_request(data_dir, "wk_test", "wr_seed_nyi");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["request", "send", "wr_seed_nyi"])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.code(1)
|
|
||||||
.stderr(contains("WebSocket request send is not implemented yet in yaak-cli"));
|
|
||||||
}
|
|
||||||
@@ -1,79 +0,0 @@
|
|||||||
mod common;
|
|
||||||
|
|
||||||
use common::http_server::TestHttpServer;
|
|
||||||
use common::{cli_cmd, query_manager, seed_folder, seed_workspace};
|
|
||||||
use predicates::str::contains;
|
|
||||||
use tempfile::TempDir;
|
|
||||||
use yaak_models::models::HttpRequest;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn top_level_send_workspace_sends_http_requests_and_prints_summary() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
|
|
||||||
let server = TestHttpServer::spawn_ok("workspace bulk send");
|
|
||||||
let request = HttpRequest {
|
|
||||||
id: "rq_workspace_send".to_string(),
|
|
||||||
workspace_id: "wk_test".to_string(),
|
|
||||||
name: "Workspace Send".to_string(),
|
|
||||||
method: "GET".to_string(),
|
|
||||||
url: server.url.clone(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.upsert_http_request(&request, &UpdateSource::Sync)
|
|
||||||
.expect("Failed to seed workspace request");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["send", "wk_test"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("workspace bulk send"))
|
|
||||||
.stdout(contains("Send summary: 1 succeeded, 0 failed"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn top_level_send_folder_sends_http_requests_and_prints_summary() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
seed_workspace(data_dir, "wk_test");
|
|
||||||
seed_folder(data_dir, "wk_test", "fl_test");
|
|
||||||
|
|
||||||
let server = TestHttpServer::spawn_ok("folder bulk send");
|
|
||||||
let request = HttpRequest {
|
|
||||||
id: "rq_folder_send".to_string(),
|
|
||||||
workspace_id: "wk_test".to_string(),
|
|
||||||
folder_id: Some("fl_test".to_string()),
|
|
||||||
name: "Folder Send".to_string(),
|
|
||||||
method: "GET".to_string(),
|
|
||||||
url: server.url.clone(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
query_manager(data_dir)
|
|
||||||
.connect()
|
|
||||||
.upsert_http_request(&request, &UpdateSource::Sync)
|
|
||||||
.expect("Failed to seed folder request");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["send", "fl_test"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("folder bulk send"))
|
|
||||||
.stdout(contains("Send summary: 1 succeeded, 0 failed"));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn top_level_send_unknown_id_fails_with_clear_error() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["send", "does_not_exist"])
|
|
||||||
.assert()
|
|
||||||
.failure()
|
|
||||||
.code(1)
|
|
||||||
.stderr(contains("Could not resolve ID 'does_not_exist' as request, folder, or workspace"));
|
|
||||||
}
|
|
||||||
@@ -1,75 +0,0 @@
|
|||||||
mod common;
|
|
||||||
|
|
||||||
use common::{cli_cmd, parse_created_id, query_manager};
|
|
||||||
use predicates::str::contains;
|
|
||||||
use tempfile::TempDir;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn create_show_delete_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
let create_assert =
|
|
||||||
cli_cmd(data_dir).args(["workspace", "create", "--name", "WS One"]).assert().success();
|
|
||||||
let workspace_id = parse_created_id(&create_assert.get_output().stdout, "workspace create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["workspace", "show", &workspace_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("\"id\": \"{workspace_id}\"")))
|
|
||||||
.stdout(contains("\"name\": \"WS One\""));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["workspace", "delete", &workspace_id, "--yes"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Deleted workspace: {workspace_id}")));
|
|
||||||
|
|
||||||
assert!(query_manager(data_dir).connect().get_workspace(&workspace_id).is_err());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn json_create_and_update_merge_patch_round_trip() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
let create_assert = cli_cmd(data_dir)
|
|
||||||
.args(["workspace", "create", r#"{"name":"Json Workspace"}"#])
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
let workspace_id = parse_created_id(&create_assert.get_output().stdout, "workspace create");
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args([
|
|
||||||
"workspace",
|
|
||||||
"update",
|
|
||||||
&format!(r#"{{"id":"{}","description":"Updated via JSON"}}"#, workspace_id),
|
|
||||||
])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains(format!("Updated workspace: {workspace_id}")));
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["workspace", "show", &workspace_id])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"name\": \"Json Workspace\""))
|
|
||||||
.stdout(contains("\"description\": \"Updated via JSON\""));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn workspace_schema_outputs_json_schema() {
|
|
||||||
let temp_dir = TempDir::new().expect("Failed to create temp dir");
|
|
||||||
let data_dir = temp_dir.path();
|
|
||||||
|
|
||||||
cli_cmd(data_dir)
|
|
||||||
.args(["workspace", "schema"])
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(contains("\"type\":\"object\""))
|
|
||||||
.stdout(contains("\"x-yaak-agent-hints\""))
|
|
||||||
.stdout(contains("\"templateVariableSyntax\":\"${[ my_var ]}\""))
|
|
||||||
.stdout(contains("\"templateFunctionSyntax\":\"${[ namespace.my_func(a='aaa',b='bbb') ]}\""))
|
|
||||||
.stdout(contains("\"name\""));
|
|
||||||
}
|
|
||||||
@@ -1,78 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "yaak-app"
|
|
||||||
version = "0.0.0"
|
|
||||||
edition = "2024"
|
|
||||||
authors = ["Gregory Schier"]
|
|
||||||
publish = false
|
|
||||||
|
|
||||||
# Produce a library for mobile support
|
|
||||||
[lib]
|
|
||||||
name = "tauri_app_lib"
|
|
||||||
crate-type = ["staticlib", "cdylib", "lib"]
|
|
||||||
|
|
||||||
[features]
|
|
||||||
cargo-clippy = []
|
|
||||||
default = []
|
|
||||||
updater = []
|
|
||||||
license = ["yaak-license"]
|
|
||||||
|
|
||||||
[build-dependencies]
|
|
||||||
tauri-build = { version = "2.5.3", features = [] }
|
|
||||||
|
|
||||||
[target.'cfg(target_os = "linux")'.dependencies]
|
|
||||||
openssl-sys = { version = "0.9.105", features = ["vendored"] } # For Ubuntu installation to work
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
charset = "0.1.5"
|
|
||||||
chrono = { workspace = true, features = ["serde"] }
|
|
||||||
cookie = "0.18.1"
|
|
||||||
eventsource-client = { git = "https://github.com/yaakapp/rust-eventsource-client", version = "0.14.0" }
|
|
||||||
http = { version = "1.2.0", default-features = false }
|
|
||||||
log = { workspace = true }
|
|
||||||
md5 = "0.8.0"
|
|
||||||
r2d2 = "0.8.10"
|
|
||||||
r2d2_sqlite = "0.25.0"
|
|
||||||
mime_guess = "2.0.5"
|
|
||||||
rand = "0.9.0"
|
|
||||||
reqwest = { workspace = true, features = ["multipart", "gzip", "brotli", "deflate", "json", "rustls-tls-manual-roots-no-provider", "socks", "http2"] }
|
|
||||||
serde = { workspace = true, features = ["derive"] }
|
|
||||||
serde_json = { workspace = true, features = ["raw_value"] }
|
|
||||||
tauri = { workspace = true, features = ["devtools", "protocol-asset"] }
|
|
||||||
tauri-plugin-clipboard-manager = "2.3.2"
|
|
||||||
tauri-plugin-deep-link = "2.4.5"
|
|
||||||
tauri-plugin-dialog = { workspace = true }
|
|
||||||
tauri-plugin-fs = "2.4.4"
|
|
||||||
tauri-plugin-log = { version = "2.7.1", features = ["colored"] }
|
|
||||||
tauri-plugin-opener = "2.5.2"
|
|
||||||
tauri-plugin-os = "2.3.2"
|
|
||||||
tauri-plugin-shell = { workspace = true }
|
|
||||||
tauri-plugin-single-instance = { version = "2.3.6", features = ["deep-link"] }
|
|
||||||
tauri-plugin-updater = "2.9.0"
|
|
||||||
tauri-plugin-window-state = "2.4.1"
|
|
||||||
thiserror = { workspace = true }
|
|
||||||
tokio = { workspace = true, features = ["sync"] }
|
|
||||||
tokio-stream = "0.1.17"
|
|
||||||
tokio-tungstenite = { version = "0.26.2", default-features = false }
|
|
||||||
url = "2"
|
|
||||||
tokio-util = { version = "0.7", features = ["codec"] }
|
|
||||||
ts-rs = { workspace = true }
|
|
||||||
uuid = "1.12.1"
|
|
||||||
yaak-api = { workspace = true }
|
|
||||||
yaak-common = { workspace = true }
|
|
||||||
yaak-tauri-utils = { workspace = true }
|
|
||||||
yaak-core = { workspace = true }
|
|
||||||
yaak = { workspace = true }
|
|
||||||
yaak-crypto = { workspace = true }
|
|
||||||
yaak-fonts = { workspace = true }
|
|
||||||
yaak-git = { workspace = true }
|
|
||||||
yaak-grpc = { workspace = true }
|
|
||||||
yaak-http = { workspace = true }
|
|
||||||
yaak-license = { workspace = true, optional = true }
|
|
||||||
yaak-mac-window = { workspace = true }
|
|
||||||
yaak-models = { workspace = true }
|
|
||||||
yaak-plugins = { workspace = true }
|
|
||||||
yaak-sse = { workspace = true }
|
|
||||||
yaak-sync = { workspace = true }
|
|
||||||
yaak-templates = { workspace = true }
|
|
||||||
yaak-tls = { workspace = true }
|
|
||||||
yaak-ws = { workspace = true }
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.
|
|
||||||
|
|
||||||
export type WatchResult = { unlistenEvent: string, };
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.
|
|
||||||
|
|
||||||
export type PluginUpdateInfo = { name: string, currentVersion: string, latestVersion: string, };
|
|
||||||
|
|
||||||
export type PluginUpdateNotification = { updateCount: number, plugins: Array<PluginUpdateInfo>, };
|
|
||||||
|
|
||||||
export type UpdateInfo = { replyEventId: string, version: string, downloaded: boolean, };
|
|
||||||
|
|
||||||
export type UpdateResponse = { "type": "ack" } | { "type": "action", action: UpdateResponseAction, };
|
|
||||||
|
|
||||||
export type UpdateResponseAction = "install" | "skip";
|
|
||||||
|
|
||||||
export type WatchResult = { unlistenEvent: string, };
|
|
||||||
|
|
||||||
export type YaakNotification = { timestamp: string, timeout: number | null, id: string, title: string | null, message: string, color: string | null, action: YaakNotificationAction | null, };
|
|
||||||
|
|
||||||
export type YaakNotificationAction = { label: string, url: string, };
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.
|
|
||||||
|
|
||||||
export type PluginUpdateInfo = { name: string, currentVersion: string, latestVersion: string, };
|
|
||||||
|
|
||||||
export type PluginUpdateNotification = { updateCount: number, plugins: Array<PluginUpdateInfo>, };
|
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 2.9 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 5.6 KiB |
@@ -1,13 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
|
||||||
<plist version="1.0">
|
|
||||||
<dict>
|
|
||||||
<!-- Enable for NodeJS/V8 JIT compiler -->
|
|
||||||
<key>com.apple.security.cs.allow-unsigned-executable-memory</key>
|
|
||||||
<true/>
|
|
||||||
|
|
||||||
<!-- Allow loading plugins signed with different Team IDs (e.g., 1Password) -->
|
|
||||||
<key>com.apple.security.cs.disable-library-validation</key>
|
|
||||||
<true/>
|
|
||||||
</dict>
|
|
||||||
</plist>
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
|
||||||
<plist version="1.0">
|
|
||||||
<dict>
|
|
||||||
</dict>
|
|
||||||
</plist>
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "@yaakapp-internal/tauri",
|
|
||||||
"private": true,
|
|
||||||
"version": "1.0.0",
|
|
||||||
"main": "bindings/index.ts"
|
|
||||||
}
|
|
||||||
@@ -1,100 +0,0 @@
|
|||||||
use crate::PluginContextExt;
|
|
||||||
use crate::error::Result;
|
|
||||||
use std::sync::Arc;
|
|
||||||
use tauri::{AppHandle, Manager, Runtime, State, WebviewWindow, command};
|
|
||||||
use yaak_crypto::manager::EncryptionManager;
|
|
||||||
use yaak_models::models::HttpRequestHeader;
|
|
||||||
use yaak_models::queries::workspaces::default_headers;
|
|
||||||
use yaak_plugins::events::GetThemesResponse;
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
use yaak_plugins::native_template_functions::{
|
|
||||||
decrypt_secure_template_function, encrypt_secure_template_function,
|
|
||||||
};
|
|
||||||
|
|
||||||
/// Extension trait for accessing the EncryptionManager from Tauri Manager types.
|
|
||||||
pub trait EncryptionManagerExt<'a, R> {
|
|
||||||
fn crypto(&'a self) -> State<'a, EncryptionManager>;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, R: Runtime, M: Manager<R>> EncryptionManagerExt<'a, R> for M {
|
|
||||||
fn crypto(&'a self) -> State<'a, EncryptionManager> {
|
|
||||||
self.state::<EncryptionManager>()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_decrypt_template<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
template: &str,
|
|
||||||
) -> Result<String> {
|
|
||||||
let encryption_manager = window.app_handle().state::<EncryptionManager>();
|
|
||||||
let plugin_context = window.plugin_context();
|
|
||||||
Ok(decrypt_secure_template_function(&encryption_manager, &plugin_context, template)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_secure_template<R: Runtime>(
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
template: &str,
|
|
||||||
) -> Result<String> {
|
|
||||||
let plugin_manager = Arc::new((*app_handle.state::<PluginManager>()).clone());
|
|
||||||
let encryption_manager = Arc::new((*app_handle.state::<EncryptionManager>()).clone());
|
|
||||||
let plugin_context = window.plugin_context();
|
|
||||||
Ok(encrypt_secure_template_function(
|
|
||||||
plugin_manager,
|
|
||||||
encryption_manager,
|
|
||||||
&plugin_context,
|
|
||||||
template,
|
|
||||||
)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_get_themes<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
plugin_manager: State<'_, PluginManager>,
|
|
||||||
) -> Result<Vec<GetThemesResponse>> {
|
|
||||||
Ok(plugin_manager.get_themes(&window.plugin_context()).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_enable_encryption<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
workspace_id: &str,
|
|
||||||
) -> Result<()> {
|
|
||||||
window.crypto().ensure_workspace_key(workspace_id)?;
|
|
||||||
window.crypto().reveal_workspace_key(workspace_id)?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_reveal_workspace_key<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
workspace_id: &str,
|
|
||||||
) -> Result<String> {
|
|
||||||
Ok(window.crypto().reveal_workspace_key(workspace_id)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_set_workspace_key<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
workspace_id: &str,
|
|
||||||
key: &str,
|
|
||||||
) -> Result<()> {
|
|
||||||
window.crypto().set_human_key(workspace_id, key)?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn cmd_disable_encryption<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
workspace_id: &str,
|
|
||||||
) -> Result<()> {
|
|
||||||
window.crypto().disable_encryption(workspace_id)?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) fn cmd_default_headers() -> Vec<HttpRequestHeader> {
|
|
||||||
default_headers()
|
|
||||||
}
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
use mime_guess::{Mime, mime};
|
|
||||||
use std::path::Path;
|
|
||||||
use std::str::FromStr;
|
|
||||||
use tokio::fs;
|
|
||||||
|
|
||||||
pub async fn read_response_body(body_path: impl AsRef<Path>, content_type: &str) -> Option<String> {
|
|
||||||
let body = fs::read(body_path).await.ok()?;
|
|
||||||
let body_charset = parse_charset(content_type).unwrap_or("utf-8".to_string());
|
|
||||||
if let Some(decoder) = charset::Charset::for_label(body_charset.as_bytes()) {
|
|
||||||
let (cow, _real_encoding, _exist_replace) = decoder.decode(&body);
|
|
||||||
return cow.into_owned().into();
|
|
||||||
}
|
|
||||||
|
|
||||||
Some(String::from_utf8_lossy(&body).to_string())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_charset(content_type: &str) -> Option<String> {
|
|
||||||
let mime: Mime = Mime::from_str(content_type).ok()?;
|
|
||||||
mime.get_param(mime::CHARSET).map(|v| v.to_string())
|
|
||||||
}
|
|
||||||
@@ -1,149 +0,0 @@
|
|||||||
//! Tauri-specific extensions for yaak-git.
|
|
||||||
//!
|
|
||||||
//! This module provides the Tauri commands for git functionality.
|
|
||||||
|
|
||||||
use crate::error::Result;
|
|
||||||
use std::path::{Path, PathBuf};
|
|
||||||
use tauri::command;
|
|
||||||
use yaak_git::{
|
|
||||||
BranchDeleteResult, CloneResult, GitCommit, GitRemote, GitStatusSummary, PullResult,
|
|
||||||
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_init, git_log, git_merge_branch, git_pull, git_pull_force_reset, git_pull_merge, git_push,
|
|
||||||
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
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_checkout(dir: &Path, branch: &str, force: bool) -> Result<String> {
|
|
||||||
Ok(git_checkout_branch(dir, branch, force).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_branch(dir: &Path, branch: &str, base: Option<&str>) -> Result<()> {
|
|
||||||
Ok(git_create_branch(dir, branch, base).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_delete_branch(
|
|
||||||
dir: &Path,
|
|
||||||
branch: &str,
|
|
||||||
force: Option<bool>,
|
|
||||||
) -> Result<BranchDeleteResult> {
|
|
||||||
Ok(git_delete_branch(dir, branch, force.unwrap_or(false)).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_delete_remote_branch(dir: &Path, branch: &str) -> Result<()> {
|
|
||||||
Ok(git_delete_remote_branch(dir, branch).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_merge_branch(dir: &Path, branch: &str) -> Result<()> {
|
|
||||||
Ok(git_merge_branch(dir, branch).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_rename_branch(dir: &Path, old_name: &str, new_name: &str) -> Result<()> {
|
|
||||||
Ok(git_rename_branch(dir, old_name, new_name).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_status(dir: &Path) -> Result<GitStatusSummary> {
|
|
||||||
Ok(git_status(dir)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_log(dir: &Path) -> Result<Vec<GitCommit>> {
|
|
||||||
Ok(git_log(dir)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_initialize(dir: &Path) -> Result<()> {
|
|
||||||
Ok(git_init(dir)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_clone(url: &str, dir: &Path) -> Result<CloneResult> {
|
|
||||||
Ok(git_clone(url, dir).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_commit(dir: &Path, message: &str) -> Result<()> {
|
|
||||||
Ok(git_commit(dir, message).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_fetch_all(dir: &Path) -> Result<()> {
|
|
||||||
Ok(git_fetch_all(dir).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_push(dir: &Path) -> Result<PushResult> {
|
|
||||||
Ok(git_push(dir).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_pull(dir: &Path) -> Result<PullResult> {
|
|
||||||
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]
|
|
||||||
pub async fn cmd_git_add(dir: &Path, rela_paths: Vec<PathBuf>) -> Result<()> {
|
|
||||||
for path in rela_paths {
|
|
||||||
git_add(dir, &path)?;
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_unstage(dir: &Path, rela_paths: Vec<PathBuf>) -> Result<()> {
|
|
||||||
for path in rela_paths {
|
|
||||||
git_unstage(dir, &path)?;
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_reset_changes(dir: &Path) -> Result<()> {
|
|
||||||
Ok(git_reset_changes(dir).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_add_credential(
|
|
||||||
remote_url: &str,
|
|
||||||
username: &str,
|
|
||||||
password: &str,
|
|
||||||
) -> Result<()> {
|
|
||||||
Ok(git_add_credential(remote_url, username, password).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_remotes(dir: &Path) -> Result<Vec<GitRemote>> {
|
|
||||||
Ok(git_remotes(dir)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_add_remote(dir: &Path, name: &str, url: &str) -> Result<GitRemote> {
|
|
||||||
Ok(git_add_remote(dir, name, url)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_git_rm_remote(dir: &Path, name: &str) -> Result<()> {
|
|
||||||
Ok(git_rm_remote(dir, name)?)
|
|
||||||
}
|
|
||||||
@@ -1,98 +0,0 @@
|
|||||||
use std::collections::BTreeMap;
|
|
||||||
|
|
||||||
use crate::PluginContextExt;
|
|
||||||
use crate::error::Result;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use KeyAndValueRef::{Ascii, Binary};
|
|
||||||
use tauri::{Manager, Runtime, WebviewWindow};
|
|
||||||
use yaak_grpc::{KeyAndValueRef, MetadataMap};
|
|
||||||
use yaak_models::models::GrpcRequest;
|
|
||||||
use yaak_plugins::events::{CallHttpAuthenticationRequest, HttpHeader};
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
|
|
||||||
pub(crate) fn metadata_to_map(metadata: MetadataMap) -> BTreeMap<String, String> {
|
|
||||||
let mut entries = BTreeMap::new();
|
|
||||||
for r in metadata.iter() {
|
|
||||||
match r {
|
|
||||||
Ascii(k, v) => entries.insert(k.to_string(), v.to_str().unwrap().to_string()),
|
|
||||||
Binary(k, v) => entries.insert(k.to_string(), format!("{:?}", v)),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
entries
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn resolve_grpc_request<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
request: &GrpcRequest,
|
|
||||||
) -> Result<(GrpcRequest, String)> {
|
|
||||||
let mut new_request = request.clone();
|
|
||||||
|
|
||||||
let (authentication_type, authentication, authentication_context_id) =
|
|
||||||
window.db().resolve_auth_for_grpc_request(request)?;
|
|
||||||
new_request.authentication_type = authentication_type;
|
|
||||||
new_request.authentication = authentication;
|
|
||||||
|
|
||||||
let metadata = window.db().resolve_metadata_for_grpc_request(request)?;
|
|
||||||
new_request.metadata = metadata;
|
|
||||||
|
|
||||||
Ok((new_request, authentication_context_id))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) async fn build_metadata<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
request: &GrpcRequest,
|
|
||||||
authentication_context_id: &str,
|
|
||||||
) -> Result<BTreeMap<String, String>> {
|
|
||||||
let plugin_manager = window.state::<PluginManager>();
|
|
||||||
let mut metadata = BTreeMap::new();
|
|
||||||
|
|
||||||
// Add the rest of metadata
|
|
||||||
for h in request.metadata.clone() {
|
|
||||||
if h.name.is_empty() && h.value.is_empty() {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if !h.enabled {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
metadata.insert(h.name, h.value);
|
|
||||||
}
|
|
||||||
|
|
||||||
match request.authentication_type.clone() {
|
|
||||||
None => {
|
|
||||||
// No authentication found. Not even inherited
|
|
||||||
}
|
|
||||||
Some(authentication_type) if authentication_type == "none" => {
|
|
||||||
// Explicitly no authentication
|
|
||||||
}
|
|
||||||
Some(authentication_type) => {
|
|
||||||
let auth = request.authentication.clone();
|
|
||||||
let plugin_req = CallHttpAuthenticationRequest {
|
|
||||||
context_id: format!("{:x}", md5::compute(authentication_context_id)),
|
|
||||||
values: serde_json::from_value(serde_json::to_value(&auth)?)?,
|
|
||||||
method: "POST".to_string(),
|
|
||||||
url: request.url.clone(),
|
|
||||||
headers: metadata
|
|
||||||
.iter()
|
|
||||||
.map(|(name, value)| HttpHeader {
|
|
||||||
name: name.to_string(),
|
|
||||||
value: value.to_string(),
|
|
||||||
})
|
|
||||||
.collect(),
|
|
||||||
};
|
|
||||||
let plugin_result = plugin_manager
|
|
||||||
.call_http_authentication(
|
|
||||||
&window.plugin_context(),
|
|
||||||
&authentication_type,
|
|
||||||
plugin_req,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
for header in plugin_result.set_headers.unwrap_or_default() {
|
|
||||||
metadata.insert(header.name, header.value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(metadata)
|
|
||||||
}
|
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use chrono::{NaiveDateTime, Utc};
|
|
||||||
use log::debug;
|
|
||||||
use std::sync::OnceLock;
|
|
||||||
use tauri::{AppHandle, Runtime};
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
const NAMESPACE: &str = "analytics";
|
|
||||||
const NUM_LAUNCHES_KEY: &str = "num_launches";
|
|
||||||
const LAST_VERSION_KEY: &str = "last_tracked_version";
|
|
||||||
const PREV_VERSION_KEY: &str = "last_tracked_version_prev";
|
|
||||||
const VERSION_SINCE_KEY: &str = "last_tracked_version_since";
|
|
||||||
|
|
||||||
#[derive(Default, Debug, Clone)]
|
|
||||||
pub struct LaunchEventInfo {
|
|
||||||
pub current_version: String,
|
|
||||||
pub previous_version: String,
|
|
||||||
pub launched_after_update: bool,
|
|
||||||
pub version_since: NaiveDateTime,
|
|
||||||
pub user_since: NaiveDateTime,
|
|
||||||
pub num_launches: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
static LAUNCH_INFO: OnceLock<LaunchEventInfo> = OnceLock::new();
|
|
||||||
|
|
||||||
pub fn get_or_upsert_launch_info<R: Runtime>(app_handle: &AppHandle<R>) -> &LaunchEventInfo {
|
|
||||||
LAUNCH_INFO.get_or_init(|| {
|
|
||||||
let now = Utc::now().naive_utc();
|
|
||||||
let mut info = LaunchEventInfo {
|
|
||||||
version_since: app_handle.db().get_key_value_dte(NAMESPACE, VERSION_SINCE_KEY, now),
|
|
||||||
current_version: app_handle.package_info().version.to_string(),
|
|
||||||
user_since: app_handle.db().get_settings().created_at,
|
|
||||||
num_launches: app_handle.db().get_key_value_int(NAMESPACE, NUM_LAUNCHES_KEY, 0) + 1,
|
|
||||||
|
|
||||||
// The rest will be set below
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
app_handle
|
|
||||||
.with_tx(|tx| {
|
|
||||||
// Load the previously tracked version
|
|
||||||
let curr_db = tx.get_key_value_str(NAMESPACE, LAST_VERSION_KEY, "");
|
|
||||||
let prev_db = tx.get_key_value_str(NAMESPACE, PREV_VERSION_KEY, "");
|
|
||||||
|
|
||||||
// We just updated if the app version is different from the last tracked version we stored
|
|
||||||
if !curr_db.is_empty() && info.current_version != curr_db {
|
|
||||||
info.launched_after_update = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we just updated, track the previous version as the "previous" current version
|
|
||||||
if info.launched_after_update {
|
|
||||||
info.previous_version = curr_db.clone();
|
|
||||||
info.version_since = now;
|
|
||||||
} else {
|
|
||||||
info.previous_version = prev_db.clone();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Rotate stored versions: move previous into the "prev" slot before overwriting
|
|
||||||
let source = &UpdateSource::Background;
|
|
||||||
|
|
||||||
tx.set_key_value_str(NAMESPACE, PREV_VERSION_KEY, &info.previous_version, source);
|
|
||||||
tx.set_key_value_str(NAMESPACE, LAST_VERSION_KEY, &info.current_version, source);
|
|
||||||
tx.set_key_value_dte(NAMESPACE, VERSION_SINCE_KEY, info.version_since, source);
|
|
||||||
tx.set_key_value_int(NAMESPACE, NUM_LAUNCHES_KEY, info.num_launches, source);
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
})
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
debug!("Initialized launch info");
|
|
||||||
|
|
||||||
info
|
|
||||||
})
|
|
||||||
}
|
|
||||||
@@ -1,186 +0,0 @@
|
|||||||
use crate::PluginContextExt;
|
|
||||||
use crate::error::Error::GenericError;
|
|
||||||
use crate::error::Result;
|
|
||||||
use crate::models_ext::BlobManagerExt;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use log::warn;
|
|
||||||
use std::sync::Arc;
|
|
||||||
use std::time::Instant;
|
|
||||||
use tauri::{AppHandle, Manager, Runtime, WebviewWindow};
|
|
||||||
use tokio::sync::watch::Receiver;
|
|
||||||
use yaak::send::{SendHttpRequestWithPluginsParams, send_http_request_with_plugins};
|
|
||||||
use yaak_crypto::manager::EncryptionManager;
|
|
||||||
use yaak_http::manager::HttpConnectionManager;
|
|
||||||
use yaak_models::models::{CookieJar, Environment, HttpRequest, HttpResponse, HttpResponseState};
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
use yaak_plugins::events::PluginContext;
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
|
|
||||||
/// Context for managing response state during HTTP transactions.
|
|
||||||
/// Handles both persisted responses (stored in DB) and ephemeral responses (in-memory only).
|
|
||||||
struct ResponseContext<R: Runtime> {
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
response: HttpResponse,
|
|
||||||
update_source: UpdateSource,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<R: Runtime> ResponseContext<R> {
|
|
||||||
fn new(app_handle: AppHandle<R>, response: HttpResponse, update_source: UpdateSource) -> Self {
|
|
||||||
Self { app_handle, response, update_source }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Whether this response is persisted (has a non-empty ID)
|
|
||||||
fn is_persisted(&self) -> bool {
|
|
||||||
!self.response.id.is_empty()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Update the response state. For persisted responses, fetches from DB, applies the
|
|
||||||
/// closure, and updates the DB. For ephemeral responses, just applies the closure
|
|
||||||
/// to the in-memory response.
|
|
||||||
fn update<F>(&mut self, func: F) -> Result<()>
|
|
||||||
where
|
|
||||||
F: FnOnce(&mut HttpResponse),
|
|
||||||
{
|
|
||||||
if self.is_persisted() {
|
|
||||||
let r = self.app_handle.with_tx(|tx| {
|
|
||||||
let mut r = tx.get_http_response(&self.response.id)?;
|
|
||||||
func(&mut r);
|
|
||||||
tx.update_http_response_if_id(&r, &self.update_source)?;
|
|
||||||
Ok(r)
|
|
||||||
})?;
|
|
||||||
self.response = r;
|
|
||||||
Ok(())
|
|
||||||
} else {
|
|
||||||
func(&mut self.response);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get the current response state
|
|
||||||
fn response(&self) -> &HttpResponse {
|
|
||||||
&self.response
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn send_http_request<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
unrendered_request: &HttpRequest,
|
|
||||||
og_response: &HttpResponse,
|
|
||||||
environment: Option<Environment>,
|
|
||||||
cookie_jar: Option<CookieJar>,
|
|
||||||
cancelled_rx: &mut Receiver<bool>,
|
|
||||||
) -> Result<HttpResponse> {
|
|
||||||
send_http_request_with_context(
|
|
||||||
window,
|
|
||||||
unrendered_request,
|
|
||||||
og_response,
|
|
||||||
environment,
|
|
||||||
cookie_jar,
|
|
||||||
cancelled_rx,
|
|
||||||
&window.plugin_context(),
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn send_http_request_with_context<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
unrendered_request: &HttpRequest,
|
|
||||||
og_response: &HttpResponse,
|
|
||||||
environment: Option<Environment>,
|
|
||||||
cookie_jar: Option<CookieJar>,
|
|
||||||
cancelled_rx: &Receiver<bool>,
|
|
||||||
plugin_context: &PluginContext,
|
|
||||||
) -> Result<HttpResponse> {
|
|
||||||
let app_handle = window.app_handle().clone();
|
|
||||||
let update_source = UpdateSource::from_window_label(window.label());
|
|
||||||
let mut response_ctx =
|
|
||||||
ResponseContext::new(app_handle.clone(), og_response.clone(), update_source);
|
|
||||||
|
|
||||||
// Execute the inner send logic and handle errors consistently
|
|
||||||
let start = Instant::now();
|
|
||||||
let result = send_http_request_inner(
|
|
||||||
window,
|
|
||||||
unrendered_request,
|
|
||||||
environment,
|
|
||||||
cookie_jar,
|
|
||||||
cancelled_rx,
|
|
||||||
plugin_context,
|
|
||||||
&mut response_ctx,
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
|
|
||||||
match result {
|
|
||||||
Ok(response) => Ok(response),
|
|
||||||
Err(e) => {
|
|
||||||
let error = e.to_string();
|
|
||||||
let elapsed = start.elapsed().as_millis() as i32;
|
|
||||||
warn!("Failed to send request: {error:?}");
|
|
||||||
let _ = response_ctx.update(|r| {
|
|
||||||
r.state = HttpResponseState::Closed;
|
|
||||||
r.elapsed = elapsed;
|
|
||||||
if r.elapsed_headers == 0 {
|
|
||||||
r.elapsed_headers = elapsed;
|
|
||||||
}
|
|
||||||
r.error = Some(error);
|
|
||||||
});
|
|
||||||
Ok(response_ctx.response().clone())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn send_http_request_inner<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
unrendered_request: &HttpRequest,
|
|
||||||
environment: Option<Environment>,
|
|
||||||
cookie_jar: Option<CookieJar>,
|
|
||||||
cancelled_rx: &Receiver<bool>,
|
|
||||||
plugin_context: &PluginContext,
|
|
||||||
response_ctx: &mut ResponseContext<R>,
|
|
||||||
) -> Result<HttpResponse> {
|
|
||||||
let app_handle = window.app_handle().clone();
|
|
||||||
let plugin_manager = Arc::new((*app_handle.state::<PluginManager>()).clone());
|
|
||||||
let encryption_manager = Arc::new((*app_handle.state::<EncryptionManager>()).clone());
|
|
||||||
let connection_manager = app_handle.state::<HttpConnectionManager>();
|
|
||||||
let environment_id = environment.map(|e| e.id);
|
|
||||||
let cookie_jar_id = cookie_jar.as_ref().map(|jar| jar.id.clone());
|
|
||||||
|
|
||||||
let response_dir = app_handle.path().app_data_dir()?.join("responses");
|
|
||||||
let result = send_http_request_with_plugins(SendHttpRequestWithPluginsParams {
|
|
||||||
query_manager: app_handle.db_manager().inner(),
|
|
||||||
blob_manager: app_handle.blob_manager().inner(),
|
|
||||||
request: unrendered_request.clone(),
|
|
||||||
environment_id: environment_id.as_deref(),
|
|
||||||
update_source: response_ctx.update_source.clone(),
|
|
||||||
cookie_jar_id,
|
|
||||||
response_dir: &response_dir,
|
|
||||||
emit_events_to: None,
|
|
||||||
emit_response_body_chunks_to: None,
|
|
||||||
existing_response: Some(response_ctx.response().clone()),
|
|
||||||
plugin_manager,
|
|
||||||
encryption_manager,
|
|
||||||
plugin_context,
|
|
||||||
cancelled_rx: Some(cancelled_rx.clone()),
|
|
||||||
connection_manager: Some(connection_manager.inner()),
|
|
||||||
})
|
|
||||||
.await
|
|
||||||
.map_err(|e| GenericError(e.to_string()))?;
|
|
||||||
|
|
||||||
Ok(result.response)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn resolve_http_request<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
request: &HttpRequest,
|
|
||||||
) -> Result<(HttpRequest, String)> {
|
|
||||||
let mut new_request = request.clone();
|
|
||||||
|
|
||||||
let (authentication_type, authentication, authentication_context_id) =
|
|
||||||
window.db().resolve_auth_for_http_request(request)?;
|
|
||||||
new_request.authentication_type = authentication_type;
|
|
||||||
new_request.authentication = authentication;
|
|
||||||
|
|
||||||
let headers = window.db().resolve_headers_for_http_request(request)?;
|
|
||||||
new_request.headers = headers;
|
|
||||||
|
|
||||||
Ok((new_request, authentication_context_id))
|
|
||||||
}
|
|
||||||
@@ -1,129 +0,0 @@
|
|||||||
use crate::PluginContextExt;
|
|
||||||
use crate::error::Result;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use log::info;
|
|
||||||
use std::collections::BTreeMap;
|
|
||||||
use std::fs::read_to_string;
|
|
||||||
use tauri::{Manager, Runtime, WebviewWindow};
|
|
||||||
use yaak_core::WorkspaceContext;
|
|
||||||
use yaak_models::models::{
|
|
||||||
Environment, Folder, GrpcRequest, HttpRequest, WebsocketRequest, Workspace,
|
|
||||||
};
|
|
||||||
use yaak_models::util::{BatchUpsertResult, UpdateSource, maybe_gen_id, maybe_gen_id_opt};
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
use yaak_tauri_utils::window::WorkspaceWindowTrait;
|
|
||||||
|
|
||||||
pub(crate) async fn import_data<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
file_path: &str,
|
|
||||||
) -> Result<BatchUpsertResult> {
|
|
||||||
let plugin_manager = window.state::<PluginManager>();
|
|
||||||
let file =
|
|
||||||
read_to_string(file_path).unwrap_or_else(|_| panic!("Unable to read file {}", file_path));
|
|
||||||
let file_contents = file.as_str();
|
|
||||||
let import_result = plugin_manager.import_data(&window.plugin_context(), file_contents).await?;
|
|
||||||
|
|
||||||
let mut id_map: BTreeMap<String, String> = BTreeMap::new();
|
|
||||||
|
|
||||||
// Create WorkspaceContext from window
|
|
||||||
let ctx = WorkspaceContext {
|
|
||||||
workspace_id: window.workspace_id(),
|
|
||||||
environment_id: window.environment_id(),
|
|
||||||
cookie_jar_id: window.cookie_jar_id(),
|
|
||||||
request_id: None,
|
|
||||||
};
|
|
||||||
|
|
||||||
let resources = import_result.resources;
|
|
||||||
|
|
||||||
let workspaces: Vec<Workspace> = resources
|
|
||||||
.workspaces
|
|
||||||
.into_iter()
|
|
||||||
.map(|mut v| {
|
|
||||||
v.id = maybe_gen_id::<Workspace>(&ctx, v.id.as_str(), &mut id_map);
|
|
||||||
v
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let environments: Vec<Environment> = resources
|
|
||||||
.environments
|
|
||||||
.into_iter()
|
|
||||||
.map(|mut v| {
|
|
||||||
v.id = maybe_gen_id::<Environment>(&ctx, v.id.as_str(), &mut id_map);
|
|
||||||
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
||||||
match (v.parent_model.as_str(), v.parent_id.clone().as_deref()) {
|
|
||||||
("folder", Some(parent_id)) => {
|
|
||||||
v.parent_id = Some(maybe_gen_id::<Folder>(&ctx, &parent_id, &mut id_map));
|
|
||||||
}
|
|
||||||
("", _) => {
|
|
||||||
// Fix any empty ones
|
|
||||||
v.parent_model = "workspace".to_string();
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
// Parent ID only required for the folder case
|
|
||||||
v.parent_id = None;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
v
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let folders: Vec<Folder> = resources
|
|
||||||
.folders
|
|
||||||
.into_iter()
|
|
||||||
.map(|mut v| {
|
|
||||||
v.id = maybe_gen_id::<Folder>(&ctx, v.id.as_str(), &mut id_map);
|
|
||||||
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
||||||
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
||||||
v
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let http_requests: Vec<HttpRequest> = resources
|
|
||||||
.http_requests
|
|
||||||
.into_iter()
|
|
||||||
.map(|mut v| {
|
|
||||||
v.id = maybe_gen_id::<HttpRequest>(&ctx, v.id.as_str(), &mut id_map);
|
|
||||||
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
||||||
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
||||||
v
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let grpc_requests: Vec<GrpcRequest> = resources
|
|
||||||
.grpc_requests
|
|
||||||
.into_iter()
|
|
||||||
.map(|mut v| {
|
|
||||||
v.id = maybe_gen_id::<GrpcRequest>(&ctx, v.id.as_str(), &mut id_map);
|
|
||||||
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
||||||
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
||||||
v
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let websocket_requests: Vec<WebsocketRequest> = resources
|
|
||||||
.websocket_requests
|
|
||||||
.into_iter()
|
|
||||||
.map(|mut v| {
|
|
||||||
v.id = maybe_gen_id::<WebsocketRequest>(&ctx, v.id.as_str(), &mut id_map);
|
|
||||||
v.workspace_id = maybe_gen_id::<Workspace>(&ctx, v.workspace_id.as_str(), &mut id_map);
|
|
||||||
v.folder_id = maybe_gen_id_opt::<Folder>(&ctx, v.folder_id, &mut id_map);
|
|
||||||
v
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
info!("Importing data");
|
|
||||||
|
|
||||||
let upserted = window.with_tx(|tx| {
|
|
||||||
tx.batch_upsert(
|
|
||||||
workspaces,
|
|
||||||
environments,
|
|
||||||
folders,
|
|
||||||
http_requests,
|
|
||||||
grpc_requests,
|
|
||||||
websocket_requests,
|
|
||||||
&UpdateSource::Import,
|
|
||||||
)
|
|
||||||
})?;
|
|
||||||
|
|
||||||
Ok(upserted)
|
|
||||||
}
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,373 +0,0 @@
|
|||||||
//! Tauri-specific extensions for yaak-models.
|
|
||||||
//!
|
|
||||||
//! This module provides the Tauri plugin initialization and extension traits
|
|
||||||
//! that allow accessing QueryManager and BlobManager from Tauri's Manager types.
|
|
||||||
|
|
||||||
use chrono::Utc;
|
|
||||||
use log::error;
|
|
||||||
use std::time::Duration;
|
|
||||||
use tauri::plugin::TauriPlugin;
|
|
||||||
use tauri::{Emitter, Manager, Runtime, State};
|
|
||||||
use tauri_plugin_dialog::{DialogExt, MessageDialogKind};
|
|
||||||
use yaak_models::blob_manager::BlobManager;
|
|
||||||
use yaak_models::db_context::DbContext;
|
|
||||||
use yaak_models::error::Result;
|
|
||||||
use yaak_models::models::{AnyModel, GraphQlIntrospection, GrpcEvent, Settings, WebsocketEvent};
|
|
||||||
use yaak_models::query_manager::QueryManager;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
const MODEL_CHANGES_RETENTION_HOURS: i64 = 1;
|
|
||||||
const MODEL_CHANGES_POLL_INTERVAL_MS: u64 = 1000;
|
|
||||||
const MODEL_CHANGES_POLL_BATCH_SIZE: usize = 200;
|
|
||||||
|
|
||||||
struct ModelChangeCursor {
|
|
||||||
created_at: String,
|
|
||||||
id: i64,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ModelChangeCursor {
|
|
||||||
fn from_launch_time() -> Self {
|
|
||||||
Self {
|
|
||||||
created_at: Utc::now().naive_utc().format("%Y-%m-%d %H:%M:%S%.3f").to_string(),
|
|
||||||
id: 0,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn drain_model_changes_batch<R: Runtime>(
|
|
||||||
query_manager: &QueryManager,
|
|
||||||
app_handle: &tauri::AppHandle<R>,
|
|
||||||
cursor: &mut ModelChangeCursor,
|
|
||||||
) -> bool {
|
|
||||||
let changes = match query_manager.connect().list_model_changes_since(
|
|
||||||
&cursor.created_at,
|
|
||||||
cursor.id,
|
|
||||||
MODEL_CHANGES_POLL_BATCH_SIZE,
|
|
||||||
) {
|
|
||||||
Ok(changes) => changes,
|
|
||||||
Err(err) => {
|
|
||||||
error!("Failed to poll model_changes rows: {err:?}");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if changes.is_empty() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
let fetched_count = changes.len();
|
|
||||||
for change in changes {
|
|
||||||
cursor.created_at = change.created_at;
|
|
||||||
cursor.id = change.id;
|
|
||||||
|
|
||||||
// Local window-originated writes are forwarded immediately from the
|
|
||||||
// in-memory model event channel.
|
|
||||||
if matches!(change.payload.update_source, UpdateSource::Window { .. }) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if let Err(err) = app_handle.emit("model_write", change.payload) {
|
|
||||||
error!("Failed to emit model_write event: {err:?}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fetched_count == MODEL_CHANGES_POLL_BATCH_SIZE
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn run_model_change_poller<R: Runtime>(
|
|
||||||
query_manager: QueryManager,
|
|
||||||
app_handle: tauri::AppHandle<R>,
|
|
||||||
mut cursor: ModelChangeCursor,
|
|
||||||
) {
|
|
||||||
loop {
|
|
||||||
while drain_model_changes_batch(&query_manager, &app_handle, &mut cursor) {}
|
|
||||||
tokio::time::sleep(Duration::from_millis(MODEL_CHANGES_POLL_INTERVAL_MS)).await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Extension trait for accessing the QueryManager from Tauri Manager types.
|
|
||||||
pub trait QueryManagerExt<'a, R> {
|
|
||||||
fn db_manager(&'a self) -> State<'a, QueryManager>;
|
|
||||||
fn db(&'a self) -> DbContext<'a>;
|
|
||||||
fn with_tx<F, T>(&'a self, func: F) -> Result<T>
|
|
||||||
where
|
|
||||||
F: FnOnce(&DbContext) -> Result<T>;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, R: Runtime, M: Manager<R>> QueryManagerExt<'a, R> for M {
|
|
||||||
fn db_manager(&'a self) -> State<'a, QueryManager> {
|
|
||||||
self.state::<QueryManager>()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn db(&'a self) -> DbContext<'a> {
|
|
||||||
let qm = self.state::<QueryManager>();
|
|
||||||
qm.inner().connect()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn with_tx<F, T>(&'a self, func: F) -> Result<T>
|
|
||||||
where
|
|
||||||
F: FnOnce(&DbContext) -> Result<T>,
|
|
||||||
{
|
|
||||||
let qm = self.state::<QueryManager>();
|
|
||||||
qm.inner().with_tx(func)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Extension trait for accessing the BlobManager from Tauri Manager types.
|
|
||||||
pub trait BlobManagerExt<'a, R> {
|
|
||||||
fn blob_manager(&'a self) -> State<'a, BlobManager>;
|
|
||||||
fn blobs(&'a self) -> yaak_models::blob_manager::BlobContext;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, R: Runtime, M: Manager<R>> BlobManagerExt<'a, R> for M {
|
|
||||||
fn blob_manager(&'a self) -> State<'a, BlobManager> {
|
|
||||||
self.state::<BlobManager>()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn blobs(&'a self) -> yaak_models::blob_manager::BlobContext {
|
|
||||||
let manager = self.state::<BlobManager>();
|
|
||||||
manager.inner().connect()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Commands for yaak-models
|
|
||||||
use tauri::WebviewWindow;
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_upsert<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
model: AnyModel,
|
|
||||||
) -> Result<String> {
|
|
||||||
use yaak_models::error::Error::GenericError;
|
|
||||||
|
|
||||||
let db = window.db();
|
|
||||||
let blobs = window.blob_manager();
|
|
||||||
let source = &UpdateSource::from_window_label(window.label());
|
|
||||||
let id = match model {
|
|
||||||
AnyModel::CookieJar(m) => db.upsert_cookie_jar(&m, source)?.id,
|
|
||||||
AnyModel::Environment(m) => db.upsert_environment(&m, source)?.id,
|
|
||||||
AnyModel::Folder(m) => db.upsert_folder(&m, source)?.id,
|
|
||||||
AnyModel::GrpcRequest(m) => db.upsert_grpc_request(&m, source)?.id,
|
|
||||||
AnyModel::HttpRequest(m) => db.upsert_http_request(&m, source)?.id,
|
|
||||||
AnyModel::HttpResponse(m) => db.upsert_http_response(&m, source, &blobs)?.id,
|
|
||||||
AnyModel::KeyValue(m) => db.upsert_key_value(&m, source)?.id,
|
|
||||||
AnyModel::Plugin(m) => db.upsert_plugin(&m, source)?.id,
|
|
||||||
AnyModel::Settings(m) => db.upsert_settings(&m, source)?.id,
|
|
||||||
AnyModel::WebsocketRequest(m) => db.upsert_websocket_request(&m, source)?.id,
|
|
||||||
AnyModel::Workspace(m) => db.upsert_workspace(&m, source)?.id,
|
|
||||||
AnyModel::WorkspaceMeta(m) => db.upsert_workspace_meta(&m, source)?.id,
|
|
||||||
a => return Err(GenericError(format!("Cannot upsert AnyModel {a:?})"))),
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(id)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_delete<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
model: AnyModel,
|
|
||||||
) -> Result<String> {
|
|
||||||
use yaak_models::error::Error::GenericError;
|
|
||||||
|
|
||||||
let blobs = window.blob_manager();
|
|
||||||
// Use transaction for deletions because it might recurse
|
|
||||||
window.with_tx(|tx| {
|
|
||||||
let source = &UpdateSource::from_window_label(window.label());
|
|
||||||
let id = match model {
|
|
||||||
AnyModel::CookieJar(m) => tx.delete_cookie_jar(&m, source)?.id,
|
|
||||||
AnyModel::Environment(m) => tx.delete_environment(&m, source)?.id,
|
|
||||||
AnyModel::Folder(m) => tx.delete_folder(&m, source)?.id,
|
|
||||||
AnyModel::GrpcConnection(m) => tx.delete_grpc_connection(&m, source)?.id,
|
|
||||||
AnyModel::GrpcRequest(m) => tx.delete_grpc_request(&m, source)?.id,
|
|
||||||
AnyModel::HttpRequest(m) => tx.delete_http_request(&m, source)?.id,
|
|
||||||
AnyModel::HttpResponse(m) => tx.delete_http_response(&m, source, &blobs)?.id,
|
|
||||||
AnyModel::Plugin(m) => tx.delete_plugin(&m, source)?.id,
|
|
||||||
AnyModel::WebsocketConnection(m) => tx.delete_websocket_connection(&m, source)?.id,
|
|
||||||
AnyModel::WebsocketRequest(m) => tx.delete_websocket_request(&m, source)?.id,
|
|
||||||
AnyModel::Workspace(m) => tx.delete_workspace(&m, source)?.id,
|
|
||||||
a => return Err(GenericError(format!("Cannot delete AnyModel {a:?})"))),
|
|
||||||
};
|
|
||||||
Ok(id)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_duplicate<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
model: AnyModel,
|
|
||||||
) -> Result<String> {
|
|
||||||
use yaak_models::error::Error::GenericError;
|
|
||||||
|
|
||||||
// Use transaction for duplications because it might recurse
|
|
||||||
window.with_tx(|tx| {
|
|
||||||
let source = &UpdateSource::from_window_label(window.label());
|
|
||||||
let id = match model {
|
|
||||||
AnyModel::Environment(m) => tx.duplicate_environment(&m, source)?.id,
|
|
||||||
AnyModel::Folder(m) => tx.duplicate_folder(&m, source)?.id,
|
|
||||||
AnyModel::GrpcRequest(m) => tx.duplicate_grpc_request(&m, source)?.id,
|
|
||||||
AnyModel::HttpRequest(m) => tx.duplicate_http_request(&m, source)?.id,
|
|
||||||
AnyModel::WebsocketRequest(m) => tx.duplicate_websocket_request(&m, source)?.id,
|
|
||||||
a => return Err(GenericError(format!("Cannot duplicate AnyModel {a:?})"))),
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(id)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_websocket_events<R: Runtime>(
|
|
||||||
app_handle: tauri::AppHandle<R>,
|
|
||||||
connection_id: &str,
|
|
||||||
) -> Result<Vec<WebsocketEvent>> {
|
|
||||||
Ok(app_handle.db().list_websocket_events(connection_id)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_grpc_events<R: Runtime>(
|
|
||||||
app_handle: tauri::AppHandle<R>,
|
|
||||||
connection_id: &str,
|
|
||||||
) -> Result<Vec<GrpcEvent>> {
|
|
||||||
Ok(app_handle.db().list_grpc_events(connection_id)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_get_settings<R: Runtime>(app_handle: tauri::AppHandle<R>) -> Result<Settings> {
|
|
||||||
Ok(app_handle.db().get_settings())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_get_graphql_introspection<R: Runtime>(
|
|
||||||
app_handle: tauri::AppHandle<R>,
|
|
||||||
request_id: &str,
|
|
||||||
) -> Result<Option<GraphQlIntrospection>> {
|
|
||||||
Ok(app_handle.db().get_graphql_introspection(request_id))
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_upsert_graphql_introspection<R: Runtime>(
|
|
||||||
app_handle: tauri::AppHandle<R>,
|
|
||||||
request_id: &str,
|
|
||||||
workspace_id: &str,
|
|
||||||
content: Option<String>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<GraphQlIntrospection> {
|
|
||||||
let source = UpdateSource::from_window_label(window.label());
|
|
||||||
Ok(app_handle.db().upsert_graphql_introspection(workspace_id, request_id, content, &source)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tauri::command]
|
|
||||||
pub(crate) fn models_workspace_models<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
workspace_id: Option<&str>,
|
|
||||||
) -> Result<String> {
|
|
||||||
let db = window.db();
|
|
||||||
let mut l: Vec<AnyModel> = Vec::new();
|
|
||||||
|
|
||||||
// Add the settings
|
|
||||||
l.push(db.get_settings().into());
|
|
||||||
|
|
||||||
// Add global models
|
|
||||||
l.append(&mut db.list_workspaces()?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_key_values()?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_plugins()?.into_iter().map(Into::into).collect());
|
|
||||||
|
|
||||||
// Add the workspace children
|
|
||||||
if let Some(wid) = workspace_id {
|
|
||||||
l.append(&mut db.list_cookie_jars(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_environments_ensure_base(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_folders(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_grpc_connections(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_grpc_requests(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_http_requests(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_http_responses(wid, None)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_websocket_connections(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_websocket_requests(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
l.append(&mut db.list_workspace_metas(wid)?.into_iter().map(Into::into).collect());
|
|
||||||
}
|
|
||||||
|
|
||||||
let j = serde_json::to_string(&l)?;
|
|
||||||
|
|
||||||
Ok(escape_str_for_webview(&j))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn escape_str_for_webview(input: &str) -> String {
|
|
||||||
input
|
|
||||||
.chars()
|
|
||||||
.map(|c| {
|
|
||||||
let code = c as u32;
|
|
||||||
// ASCII
|
|
||||||
if code <= 0x7F {
|
|
||||||
c.to_string()
|
|
||||||
// BMP characters encoded normally
|
|
||||||
} else if code < 0xFFFF {
|
|
||||||
format!("\\u{:04X}", code)
|
|
||||||
// Beyond BMP encoded a surrogate pairs
|
|
||||||
} else {
|
|
||||||
let high = ((code - 0x10000) >> 10) + 0xD800;
|
|
||||||
let low = ((code - 0x10000) & 0x3FF) + 0xDC00;
|
|
||||||
format!("\\u{:04X}\\u{:04X}", high, low)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Initialize database managers as a plugin (for initialization order).
|
|
||||||
/// Commands are in the main invoke_handler.
|
|
||||||
/// This must be registered before other plugins that depend on the database.
|
|
||||||
pub fn init<R: Runtime>() -> TauriPlugin<R> {
|
|
||||||
tauri::plugin::Builder::new("yaak-models-db")
|
|
||||||
.setup(|app_handle, _api| {
|
|
||||||
let app_path = app_handle.path().app_data_dir().unwrap();
|
|
||||||
let db_path = app_path.join("db.sqlite");
|
|
||||||
let blob_path = app_path.join("blobs.sqlite");
|
|
||||||
|
|
||||||
let (query_manager, blob_manager, rx) =
|
|
||||||
match yaak_models::init_standalone(&db_path, &blob_path) {
|
|
||||||
Ok(result) => result,
|
|
||||||
Err(e) => {
|
|
||||||
app_handle
|
|
||||||
.dialog()
|
|
||||||
.message(e.to_string())
|
|
||||||
.kind(MessageDialogKind::Error)
|
|
||||||
.blocking_show();
|
|
||||||
return Err(Box::from(e.to_string()));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let db = query_manager.connect();
|
|
||||||
if let Err(err) = db.prune_model_changes_older_than_hours(MODEL_CHANGES_RETENTION_HOURS)
|
|
||||||
{
|
|
||||||
error!("Failed to prune model_changes rows on startup: {err:?}");
|
|
||||||
}
|
|
||||||
// Only stream writes that happen after this app launch.
|
|
||||||
let cursor = ModelChangeCursor::from_launch_time();
|
|
||||||
|
|
||||||
let poll_query_manager = query_manager.clone();
|
|
||||||
|
|
||||||
app_handle.manage(query_manager);
|
|
||||||
app_handle.manage(blob_manager);
|
|
||||||
|
|
||||||
// Poll model_changes so all writers (including external CLI processes) update the UI.
|
|
||||||
let app_handle_poll = app_handle.clone();
|
|
||||||
let query_manager = poll_query_manager;
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
run_model_change_poller(query_manager, app_handle_poll, cursor).await;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Fast path for local app writes initiated by frontend windows. This keeps the
|
|
||||||
// current sync-model UX snappy, while DB polling handles external writers (CLI).
|
|
||||||
let app_handle_local = app_handle.clone();
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
for payload in rx {
|
|
||||||
if !matches!(payload.update_source, UpdateSource::Window { .. }) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if let Err(err) = app_handle_local.emit("model_write", payload) {
|
|
||||||
error!("Failed to emit local model_write event: {err:?}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
})
|
|
||||||
.build()
|
|
||||||
}
|
|
||||||
@@ -1,173 +0,0 @@
|
|||||||
use crate::error::Result;
|
|
||||||
use crate::history::get_or_upsert_launch_info;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use chrono::{DateTime, Utc};
|
|
||||||
use log::{debug, info};
|
|
||||||
use reqwest::Method;
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
use std::time::Instant;
|
|
||||||
use tauri::{AppHandle, Emitter, Manager, Runtime, WebviewWindow};
|
|
||||||
use ts_rs::TS;
|
|
||||||
use yaak_api::yaak_api_client;
|
|
||||||
use yaak_common::platform::get_os_str;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
|
|
||||||
// Check for updates every hour
|
|
||||||
const MAX_UPDATE_CHECK_SECONDS: u64 = 60 * 60;
|
|
||||||
|
|
||||||
const KV_NAMESPACE: &str = "notifications";
|
|
||||||
const KV_KEY: &str = "seen";
|
|
||||||
|
|
||||||
// Create updater struct
|
|
||||||
pub struct YaakNotifier {
|
|
||||||
last_check: Option<Instant>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize, Default, TS)]
|
|
||||||
#[serde(default, rename_all = "camelCase")]
|
|
||||||
#[ts(export, export_to = "index.ts")]
|
|
||||||
pub struct YaakNotification {
|
|
||||||
timestamp: DateTime<Utc>,
|
|
||||||
timeout: Option<f64>,
|
|
||||||
id: String,
|
|
||||||
title: Option<String>,
|
|
||||||
message: String,
|
|
||||||
color: Option<String>,
|
|
||||||
action: Option<YaakNotificationAction>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize, Default, TS)]
|
|
||||||
#[serde(default, rename_all = "camelCase")]
|
|
||||||
#[ts(export, export_to = "index.ts")]
|
|
||||||
pub struct YaakNotificationAction {
|
|
||||||
label: String,
|
|
||||||
url: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl YaakNotifier {
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self { last_check: None }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn seen<R: Runtime>(&mut self, window: &WebviewWindow<R>, id: &str) -> Result<()> {
|
|
||||||
let app_handle = window.app_handle();
|
|
||||||
let mut seen = get_kv(app_handle).await?;
|
|
||||||
seen.push(id.to_string());
|
|
||||||
debug!("Marked notification as seen {}", id);
|
|
||||||
let seen_json = serde_json::to_string(&seen)?;
|
|
||||||
window.db().set_key_value_raw(
|
|
||||||
KV_NAMESPACE,
|
|
||||||
KV_KEY,
|
|
||||||
seen_json.as_str(),
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn maybe_check<R: Runtime>(&mut self, window: &WebviewWindow<R>) -> Result<()> {
|
|
||||||
let app_handle = window.app_handle();
|
|
||||||
if let Some(i) = self.last_check
|
|
||||||
&& i.elapsed().as_secs() < MAX_UPDATE_CHECK_SECONDS
|
|
||||||
{
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
self.last_check = Some(Instant::now());
|
|
||||||
|
|
||||||
if !app_handle.db().get_settings().check_notifications {
|
|
||||||
info!("Notifications are disabled. Skipping check.");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
debug!("Checking for notifications");
|
|
||||||
|
|
||||||
#[cfg(feature = "license")]
|
|
||||||
let license_check = {
|
|
||||||
use yaak_license::{LicenseCheckStatus, check_license};
|
|
||||||
match check_license(window).await {
|
|
||||||
Ok(LicenseCheckStatus::PersonalUse { .. }) => "personal",
|
|
||||||
Ok(LicenseCheckStatus::Active { .. }) => "commercial",
|
|
||||||
Ok(LicenseCheckStatus::PastDue { .. }) => "past_due",
|
|
||||||
Ok(LicenseCheckStatus::Inactive { .. }) => "invalid_license",
|
|
||||||
Ok(LicenseCheckStatus::Trialing { .. }) => "trialing",
|
|
||||||
Ok(LicenseCheckStatus::Expired { .. }) => "expired",
|
|
||||||
Ok(LicenseCheckStatus::Error { .. }) => "error",
|
|
||||||
Err(_) => "unknown",
|
|
||||||
}
|
|
||||||
.to_string()
|
|
||||||
};
|
|
||||||
|
|
||||||
#[cfg(not(feature = "license"))]
|
|
||||||
let license_check = "disabled".to_string();
|
|
||||||
|
|
||||||
let launch_info = get_or_upsert_launch_info(app_handle);
|
|
||||||
let app_version = app_handle.package_info().version.to_string();
|
|
||||||
let req = yaak_api_client(&app_version)?
|
|
||||||
.request(Method::GET, "https://notify.yaak.app/notifications")
|
|
||||||
.query(&[
|
|
||||||
("version", &launch_info.current_version),
|
|
||||||
("version_prev", &launch_info.previous_version),
|
|
||||||
("launches", &launch_info.num_launches.to_string()),
|
|
||||||
("installed", &launch_info.user_since.format("%Y-%m-%d").to_string()),
|
|
||||||
("license", &license_check),
|
|
||||||
("updates", &get_updater_status(app_handle).to_string()),
|
|
||||||
("platform", &get_os_str().to_string()),
|
|
||||||
]);
|
|
||||||
let resp = req.send().await?;
|
|
||||||
if resp.status() != 200 {
|
|
||||||
debug!("Skipping notification status code {}", resp.status());
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
for notification in resp.json::<Vec<YaakNotification>>().await? {
|
|
||||||
let seen = get_kv(app_handle).await?;
|
|
||||||
if seen.contains(¬ification.id) {
|
|
||||||
debug!("Already seen notification {}", notification.id);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
debug!("Got notification {:?}", notification);
|
|
||||||
|
|
||||||
let _ = app_handle.emit_to(window.label(), "notification", notification.clone());
|
|
||||||
break; // Only show one notification
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn get_kv<R: Runtime>(app_handle: &AppHandle<R>) -> Result<Vec<String>> {
|
|
||||||
match app_handle.db().get_key_value_raw("notifications", "seen") {
|
|
||||||
None => Ok(Vec::new()),
|
|
||||||
Some(v) => Ok(serde_json::from_str(&v.value)?),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(unused)]
|
|
||||||
fn get_updater_status<R: Runtime>(app_handle: &AppHandle<R>) -> &'static str {
|
|
||||||
#[cfg(not(feature = "updater"))]
|
|
||||||
{
|
|
||||||
// Updater is not enabled as a Rust feature
|
|
||||||
return "missing";
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(all(feature = "updater", target_os = "linux"))]
|
|
||||||
{
|
|
||||||
let settings = app_handle.db().get_settings();
|
|
||||||
if !settings.autoupdate {
|
|
||||||
// Updates are explicitly disabled
|
|
||||||
"disabled"
|
|
||||||
} else if std::env::var("APPIMAGE").is_err() {
|
|
||||||
// Updates are enabled, but unsupported
|
|
||||||
"unsupported"
|
|
||||||
} else {
|
|
||||||
// Updates are enabled and supported
|
|
||||||
"enabled"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(all(feature = "updater", not(target_os = "linux")))]
|
|
||||||
{
|
|
||||||
let settings = app_handle.db().get_settings();
|
|
||||||
if settings.autoupdate { "enabled" } else { "disabled" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,528 +0,0 @@
|
|||||||
use crate::error::Result;
|
|
||||||
use crate::http_request::send_http_request_with_context;
|
|
||||||
use crate::models_ext::BlobManagerExt;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use crate::render::{render_grpc_request, render_http_request, render_json_value};
|
|
||||||
use crate::window::{CreateWindowConfig, create_window};
|
|
||||||
use crate::{
|
|
||||||
call_frontend, cookie_jar_from_window, environment_from_window, get_window_from_plugin_context,
|
|
||||||
workspace_from_window,
|
|
||||||
};
|
|
||||||
use chrono::Utc;
|
|
||||||
use cookie::Cookie;
|
|
||||||
use log::error;
|
|
||||||
use std::sync::Arc;
|
|
||||||
use tauri::{AppHandle, Emitter, Listener, Manager, Runtime};
|
|
||||||
use tauri_plugin_clipboard_manager::ClipboardExt;
|
|
||||||
use tauri_plugin_opener::OpenerExt;
|
|
||||||
use yaak::plugin_events::{
|
|
||||||
GroupedPluginEvent, HostRequest, SharedPluginEventContext, handle_shared_plugin_event,
|
|
||||||
};
|
|
||||||
use yaak_crypto::manager::EncryptionManager;
|
|
||||||
use yaak_models::models::{AnyModel, HttpResponse, Plugin};
|
|
||||||
use yaak_models::queries::any_request::AnyRequest;
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
use yaak_plugins::error::Error::PluginErr;
|
|
||||||
use yaak_plugins::events::{
|
|
||||||
Color, EmptyPayload, ErrorResponse, FindHttpResponsesResponse, GetCookieValueResponse, Icon,
|
|
||||||
InternalEvent, InternalEventPayload, ListCookieNamesResponse, ListOpenWorkspacesResponse,
|
|
||||||
RenderGrpcRequestResponse, RenderHttpRequestResponse, SendHttpRequestResponse,
|
|
||||||
ShowToastRequest, TemplateRenderResponse, WindowInfoResponse, WindowNavigateEvent,
|
|
||||||
WorkspaceInfo,
|
|
||||||
};
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
use yaak_plugins::plugin_handle::PluginHandle;
|
|
||||||
use yaak_plugins::template_callback::PluginTemplateCallback;
|
|
||||||
use yaak_tauri_utils::window::WorkspaceWindowTrait;
|
|
||||||
use yaak_templates::{RenderErrorBehavior, RenderOptions};
|
|
||||||
|
|
||||||
pub(crate) async fn handle_plugin_event<R: Runtime>(
|
|
||||||
app_handle: &AppHandle<R>,
|
|
||||||
event: &InternalEvent,
|
|
||||||
plugin_handle: &PluginHandle,
|
|
||||||
) -> Result<Option<InternalEventPayload>> {
|
|
||||||
// log::debug!("Got event to app {event:?}");
|
|
||||||
let plugin_context = event.context.to_owned();
|
|
||||||
let plugin_name = plugin_handle.info().name;
|
|
||||||
let fallback_workspace_id = plugin_context.workspace_id.clone().or_else(|| {
|
|
||||||
plugin_context
|
|
||||||
.label
|
|
||||||
.as_ref()
|
|
||||||
.and_then(|label| app_handle.get_webview_window(label))
|
|
||||||
.and_then(|window| workspace_from_window(&window).map(|workspace| workspace.id))
|
|
||||||
});
|
|
||||||
|
|
||||||
match handle_shared_plugin_event(
|
|
||||||
app_handle.db_manager().inner(),
|
|
||||||
&event.payload,
|
|
||||||
SharedPluginEventContext {
|
|
||||||
plugin_name: &plugin_name,
|
|
||||||
workspace_id: fallback_workspace_id.as_deref(),
|
|
||||||
},
|
|
||||||
) {
|
|
||||||
GroupedPluginEvent::Handled(payload) => Ok(payload),
|
|
||||||
GroupedPluginEvent::ToHandle(host_request) => {
|
|
||||||
handle_host_plugin_request(
|
|
||||||
app_handle,
|
|
||||||
event,
|
|
||||||
plugin_handle,
|
|
||||||
&plugin_context,
|
|
||||||
host_request,
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn handle_host_plugin_request<R: Runtime>(
|
|
||||||
app_handle: &AppHandle<R>,
|
|
||||||
event: &InternalEvent,
|
|
||||||
plugin_handle: &PluginHandle,
|
|
||||||
plugin_context: &yaak_plugins::events::PluginContext,
|
|
||||||
host_request: HostRequest<'_>,
|
|
||||||
) -> Result<Option<InternalEventPayload>> {
|
|
||||||
match host_request {
|
|
||||||
HostRequest::ErrorResponse(resp) => {
|
|
||||||
error!("Plugin error: {}: {:?}", resp.error, resp);
|
|
||||||
let toast_event = plugin_handle.build_event_to_send(
|
|
||||||
plugin_context,
|
|
||||||
&InternalEventPayload::ShowToastRequest(ShowToastRequest {
|
|
||||||
message: format!(
|
|
||||||
"Plugin error from {}: {}",
|
|
||||||
plugin_handle.info().name,
|
|
||||||
resp.error
|
|
||||||
),
|
|
||||||
color: Some(Color::Danger),
|
|
||||||
timeout: Some(30000),
|
|
||||||
..Default::default()
|
|
||||||
}),
|
|
||||||
None,
|
|
||||||
);
|
|
||||||
Box::pin(handle_plugin_event(app_handle, &toast_event, plugin_handle)).await
|
|
||||||
}
|
|
||||||
HostRequest::ReloadResponse(req) => {
|
|
||||||
let plugins = app_handle.db().list_plugins()?;
|
|
||||||
for plugin in plugins {
|
|
||||||
if plugin.directory != plugin_handle.dir {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
let new_plugin = Plugin { updated_at: Utc::now().naive_utc(), ..plugin };
|
|
||||||
app_handle.db().upsert_plugin(&new_plugin, &UpdateSource::Plugin)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
if !req.silent {
|
|
||||||
let info = plugin_handle.info();
|
|
||||||
let toast_event = plugin_handle.build_event_to_send(
|
|
||||||
plugin_context,
|
|
||||||
&InternalEventPayload::ShowToastRequest(ShowToastRequest {
|
|
||||||
message: format!("Reloaded plugin {}@{}", info.name, info.version),
|
|
||||||
icon: Some(Icon::Info),
|
|
||||||
timeout: Some(3000),
|
|
||||||
..Default::default()
|
|
||||||
}),
|
|
||||||
None,
|
|
||||||
);
|
|
||||||
Box::pin(handle_plugin_event(app_handle, &toast_event, plugin_handle)).await
|
|
||||||
} else {
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
HostRequest::CopyText(req) => {
|
|
||||||
app_handle.clipboard().write_text(req.text.as_str())?;
|
|
||||||
Ok(Some(InternalEventPayload::CopyTextResponse(EmptyPayload {})))
|
|
||||||
}
|
|
||||||
HostRequest::ShowToast(req) => {
|
|
||||||
match &plugin_context.label {
|
|
||||||
Some(label) => app_handle.emit_to(label, "show_toast", req)?,
|
|
||||||
None => app_handle.emit("show_toast", req)?,
|
|
||||||
};
|
|
||||||
Ok(Some(InternalEventPayload::ShowToastResponse(EmptyPayload {})))
|
|
||||||
}
|
|
||||||
HostRequest::PromptText(_) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
Ok(call_frontend(&window, event).await)
|
|
||||||
}
|
|
||||||
HostRequest::PromptForm(_) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
if event.reply_id.is_some() {
|
|
||||||
window.emit_to(window.label(), "plugin_event", event.clone())?;
|
|
||||||
Ok(None)
|
|
||||||
} else {
|
|
||||||
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();
|
|
||||||
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
let (tx, mut rx) = tokio::sync::mpsc::channel::<InternalEvent>(128);
|
|
||||||
|
|
||||||
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);
|
|
||||||
});
|
|
||||||
|
|
||||||
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)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
HostRequest::FindHttpResponses(req) => {
|
|
||||||
let http_responses = app_handle
|
|
||||||
.db()
|
|
||||||
.list_http_responses_for_request(&req.request_id, req.limit.map(|l| l as u64))
|
|
||||||
.unwrap_or_default();
|
|
||||||
Ok(Some(InternalEventPayload::FindHttpResponsesResponse(FindHttpResponsesResponse {
|
|
||||||
http_responses,
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
HostRequest::UpsertModel(req) => {
|
|
||||||
use AnyModel::*;
|
|
||||||
let model = match &req.model {
|
|
||||||
HttpRequest(m) => {
|
|
||||||
HttpRequest(app_handle.db().upsert_http_request(m, &UpdateSource::Plugin)?)
|
|
||||||
}
|
|
||||||
GrpcRequest(m) => {
|
|
||||||
GrpcRequest(app_handle.db().upsert_grpc_request(m, &UpdateSource::Plugin)?)
|
|
||||||
}
|
|
||||||
WebsocketRequest(m) => WebsocketRequest(
|
|
||||||
app_handle.db().upsert_websocket_request(m, &UpdateSource::Plugin)?,
|
|
||||||
),
|
|
||||||
Folder(m) => Folder(app_handle.db().upsert_folder(m, &UpdateSource::Plugin)?),
|
|
||||||
Environment(m) => {
|
|
||||||
Environment(app_handle.db().upsert_environment(m, &UpdateSource::Plugin)?)
|
|
||||||
}
|
|
||||||
Workspace(m) => {
|
|
||||||
Workspace(app_handle.db().upsert_workspace(m, &UpdateSource::Plugin)?)
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
return Err(PluginErr("Upsert not supported for this model type".into()).into());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(Some(InternalEventPayload::UpsertModelResponse(
|
|
||||||
yaak_plugins::events::UpsertModelResponse { model },
|
|
||||||
)))
|
|
||||||
}
|
|
||||||
HostRequest::DeleteModel(req) => {
|
|
||||||
let model = match req.model.as_str() {
|
|
||||||
"http_request" => AnyModel::HttpRequest(
|
|
||||||
app_handle.db().delete_http_request_by_id(&req.id, &UpdateSource::Plugin)?,
|
|
||||||
),
|
|
||||||
"grpc_request" => AnyModel::GrpcRequest(
|
|
||||||
app_handle.db().delete_grpc_request_by_id(&req.id, &UpdateSource::Plugin)?,
|
|
||||||
),
|
|
||||||
"websocket_request" => AnyModel::WebsocketRequest(
|
|
||||||
app_handle
|
|
||||||
.db()
|
|
||||||
.delete_websocket_request_by_id(&req.id, &UpdateSource::Plugin)?,
|
|
||||||
),
|
|
||||||
"folder" => AnyModel::Folder(
|
|
||||||
app_handle.db().delete_folder_by_id(&req.id, &UpdateSource::Plugin)?,
|
|
||||||
),
|
|
||||||
"environment" => AnyModel::Environment(
|
|
||||||
app_handle.db().delete_environment_by_id(&req.id, &UpdateSource::Plugin)?,
|
|
||||||
),
|
|
||||||
_ => {
|
|
||||||
return Err(PluginErr("Delete not supported for this model type".into()).into());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(Some(InternalEventPayload::DeleteModelResponse(
|
|
||||||
yaak_plugins::events::DeleteModelResponse { model },
|
|
||||||
)))
|
|
||||||
}
|
|
||||||
HostRequest::RenderGrpcRequest(req) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
|
|
||||||
let workspace =
|
|
||||||
workspace_from_window(&window).expect("Failed to get workspace_id from window URL");
|
|
||||||
let environment_id = environment_from_window(&window).map(|e| e.id);
|
|
||||||
let environment_chain = window.db().resolve_environments(
|
|
||||||
&workspace.id,
|
|
||||||
req.grpc_request.folder_id.as_deref(),
|
|
||||||
environment_id.as_deref(),
|
|
||||||
)?;
|
|
||||||
let plugin_manager = Arc::new((*app_handle.state::<PluginManager>()).clone());
|
|
||||||
let encryption_manager = Arc::new((*app_handle.state::<EncryptionManager>()).clone());
|
|
||||||
let cb = PluginTemplateCallback::new(
|
|
||||||
plugin_manager,
|
|
||||||
encryption_manager,
|
|
||||||
plugin_context,
|
|
||||||
req.purpose.clone(),
|
|
||||||
);
|
|
||||||
let opt = RenderOptions { error_behavior: RenderErrorBehavior::Throw };
|
|
||||||
let grpc_request =
|
|
||||||
render_grpc_request(&req.grpc_request, environment_chain, &cb, &opt).await?;
|
|
||||||
Ok(Some(InternalEventPayload::RenderGrpcRequestResponse(RenderGrpcRequestResponse {
|
|
||||||
grpc_request,
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
HostRequest::RenderHttpRequest(req) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
|
|
||||||
let workspace =
|
|
||||||
workspace_from_window(&window).expect("Failed to get workspace_id from window URL");
|
|
||||||
let environment_id = environment_from_window(&window).map(|e| e.id);
|
|
||||||
let environment_chain = window.db().resolve_environments(
|
|
||||||
&workspace.id,
|
|
||||||
req.http_request.folder_id.as_deref(),
|
|
||||||
environment_id.as_deref(),
|
|
||||||
)?;
|
|
||||||
let plugin_manager = Arc::new((*app_handle.state::<PluginManager>()).clone());
|
|
||||||
let encryption_manager = Arc::new((*app_handle.state::<EncryptionManager>()).clone());
|
|
||||||
let cb = PluginTemplateCallback::new(
|
|
||||||
plugin_manager,
|
|
||||||
encryption_manager,
|
|
||||||
plugin_context,
|
|
||||||
req.purpose.clone(),
|
|
||||||
);
|
|
||||||
let opt = &RenderOptions { error_behavior: RenderErrorBehavior::Throw };
|
|
||||||
let http_request =
|
|
||||||
render_http_request(&req.http_request, environment_chain, &cb, opt).await?;
|
|
||||||
Ok(Some(InternalEventPayload::RenderHttpRequestResponse(RenderHttpRequestResponse {
|
|
||||||
http_request,
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
HostRequest::TemplateRender(req) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
|
|
||||||
let workspace =
|
|
||||||
workspace_from_window(&window).expect("Failed to get workspace_id from window URL");
|
|
||||||
let environment_id = environment_from_window(&window).map(|e| e.id);
|
|
||||||
let folder_id = if let Some(id) = window.request_id() {
|
|
||||||
match window.db().get_any_request(&id) {
|
|
||||||
Ok(AnyRequest::HttpRequest(r)) => r.folder_id,
|
|
||||||
Ok(AnyRequest::GrpcRequest(r)) => r.folder_id,
|
|
||||||
Ok(AnyRequest::WebsocketRequest(r)) => r.folder_id,
|
|
||||||
Err(_) => None,
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
let environment_chain = window.db().resolve_environments(
|
|
||||||
&workspace.id,
|
|
||||||
folder_id.as_deref(),
|
|
||||||
environment_id.as_deref(),
|
|
||||||
)?;
|
|
||||||
let plugin_manager = Arc::new((*app_handle.state::<PluginManager>()).clone());
|
|
||||||
let encryption_manager = Arc::new((*app_handle.state::<EncryptionManager>()).clone());
|
|
||||||
let cb = PluginTemplateCallback::new(
|
|
||||||
plugin_manager,
|
|
||||||
encryption_manager,
|
|
||||||
plugin_context,
|
|
||||||
req.purpose.clone(),
|
|
||||||
);
|
|
||||||
let opt = RenderOptions { error_behavior: RenderErrorBehavior::Throw };
|
|
||||||
let data = render_json_value(req.data.clone(), environment_chain, &cb, &opt).await?;
|
|
||||||
Ok(Some(InternalEventPayload::TemplateRenderResponse(TemplateRenderResponse { data })))
|
|
||||||
}
|
|
||||||
HostRequest::SendHttpRequest(req) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
let mut http_request = req.http_request.clone();
|
|
||||||
let workspace =
|
|
||||||
workspace_from_window(&window).expect("Failed to get workspace_id from window URL");
|
|
||||||
let cookie_jar = cookie_jar_from_window(&window);
|
|
||||||
let environment = environment_from_window(&window);
|
|
||||||
|
|
||||||
if http_request.workspace_id.is_empty() {
|
|
||||||
http_request.workspace_id = workspace.id;
|
|
||||||
}
|
|
||||||
|
|
||||||
let http_response = if http_request.id.is_empty() {
|
|
||||||
HttpResponse::default()
|
|
||||||
} else {
|
|
||||||
let blobs = window.blob_manager();
|
|
||||||
window.db().upsert_http_response(
|
|
||||||
&HttpResponse {
|
|
||||||
request_id: http_request.id.clone(),
|
|
||||||
workspace_id: http_request.workspace_id.clone(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
&blobs,
|
|
||||||
)?
|
|
||||||
};
|
|
||||||
|
|
||||||
let http_response = send_http_request_with_context(
|
|
||||||
&window,
|
|
||||||
&http_request,
|
|
||||||
&http_response,
|
|
||||||
environment,
|
|
||||||
cookie_jar,
|
|
||||||
&mut tokio::sync::watch::channel(false).1,
|
|
||||||
plugin_context,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
Ok(Some(InternalEventPayload::SendHttpRequestResponse(SendHttpRequestResponse {
|
|
||||||
http_response,
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
HostRequest::OpenWindow(req) => {
|
|
||||||
let (navigation_tx, mut navigation_rx) = tokio::sync::mpsc::channel(128);
|
|
||||||
let (close_tx, mut close_rx) = tokio::sync::mpsc::channel(128);
|
|
||||||
let win_config = CreateWindowConfig {
|
|
||||||
url: &req.url,
|
|
||||||
label: &req.label,
|
|
||||||
title: &req.title.clone().unwrap_or_default(),
|
|
||||||
navigation_tx: Some(navigation_tx),
|
|
||||||
close_tx: Some(close_tx),
|
|
||||||
inner_size: req.size.clone().map(|s| (s.width, s.height)),
|
|
||||||
data_dir_key: req.data_dir_key.clone(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
if let Err(e) = create_window(app_handle, win_config) {
|
|
||||||
let error_event = plugin_handle.build_event_to_send(
|
|
||||||
plugin_context,
|
|
||||||
&InternalEventPayload::ErrorResponse(ErrorResponse {
|
|
||||||
error: format!("Failed to create window: {:?}", e),
|
|
||||||
}),
|
|
||||||
None,
|
|
||||||
);
|
|
||||||
return Box::pin(handle_plugin_event(app_handle, &error_event, plugin_handle))
|
|
||||||
.await;
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
let event_id = event.id.clone();
|
|
||||||
let plugin_handle = plugin_handle.clone();
|
|
||||||
let plugin_context = plugin_context.clone();
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
while let Some(url) = navigation_rx.recv().await {
|
|
||||||
let url = url.to_string();
|
|
||||||
let event_to_send = plugin_handle.build_event_to_send(
|
|
||||||
&plugin_context,
|
|
||||||
&InternalEventPayload::WindowNavigateEvent(WindowNavigateEvent { url }),
|
|
||||||
Some(event_id.clone()),
|
|
||||||
);
|
|
||||||
plugin_handle.send(&event_to_send).await.unwrap();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
let event_id = event.id.clone();
|
|
||||||
let plugin_handle = plugin_handle.clone();
|
|
||||||
let plugin_context = plugin_context.clone();
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
while close_rx.recv().await.is_some() {
|
|
||||||
let event_to_send = plugin_handle.build_event_to_send(
|
|
||||||
&plugin_context,
|
|
||||||
&InternalEventPayload::WindowCloseEvent,
|
|
||||||
Some(event_id.clone()),
|
|
||||||
);
|
|
||||||
plugin_handle.send(&event_to_send).await.unwrap();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
HostRequest::CloseWindow(req) => {
|
|
||||||
if let Some(window) = app_handle.webview_windows().get(&req.label) {
|
|
||||||
window.close()?;
|
|
||||||
}
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
HostRequest::OpenExternalUrl(req) => {
|
|
||||||
app_handle.opener().open_url(&req.url, None::<&str>)?;
|
|
||||||
Ok(Some(InternalEventPayload::OpenExternalUrlResponse(EmptyPayload {})))
|
|
||||||
}
|
|
||||||
HostRequest::ListOpenWorkspaces(_) => {
|
|
||||||
let mut workspaces = Vec::new();
|
|
||||||
for (_, window) in app_handle.webview_windows() {
|
|
||||||
if let Some(workspace) = workspace_from_window(&window) {
|
|
||||||
workspaces.push(WorkspaceInfo {
|
|
||||||
id: workspace.id.clone(),
|
|
||||||
name: workspace.name.clone(),
|
|
||||||
label: window.label().to_string(),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(Some(InternalEventPayload::ListOpenWorkspacesResponse(ListOpenWorkspacesResponse {
|
|
||||||
workspaces,
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
HostRequest::ListCookieNames(_) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
let names = match cookie_jar_from_window(&window) {
|
|
||||||
None => Vec::new(),
|
|
||||||
Some(j) => j
|
|
||||||
.cookies
|
|
||||||
.into_iter()
|
|
||||||
.filter_map(|c| Cookie::parse(c.raw_cookie).ok().map(|c| c.name().to_string()))
|
|
||||||
.collect(),
|
|
||||||
};
|
|
||||||
Ok(Some(InternalEventPayload::ListCookieNamesResponse(ListCookieNamesResponse {
|
|
||||||
names,
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
HostRequest::GetCookieValue(req) => {
|
|
||||||
let window = get_window_from_plugin_context(app_handle, plugin_context)?;
|
|
||||||
let value = match cookie_jar_from_window(&window) {
|
|
||||||
None => None,
|
|
||||||
Some(j) => j.cookies.into_iter().find_map(|c| match Cookie::parse(c.raw_cookie) {
|
|
||||||
Ok(c) if c.name().to_string().eq(&req.name) => {
|
|
||||||
Some(c.value_trimmed().to_string())
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
Ok(Some(InternalEventPayload::GetCookieValueResponse(GetCookieValueResponse { value })))
|
|
||||||
}
|
|
||||||
HostRequest::WindowInfo(req) => {
|
|
||||||
let w = app_handle
|
|
||||||
.get_webview_window(&req.label)
|
|
||||||
.ok_or(PluginErr(format!("Failed to find window for {}", req.label)))?;
|
|
||||||
|
|
||||||
let environment_id = environment_from_window(&w).map(|m| m.id);
|
|
||||||
let workspace_id = workspace_from_window(&w).map(|m| m.id);
|
|
||||||
let request_id =
|
|
||||||
match app_handle.db().get_any_request(&w.request_id().unwrap_or_default()) {
|
|
||||||
Ok(AnyRequest::HttpRequest(r)) => Some(r.id),
|
|
||||||
Ok(AnyRequest::WebsocketRequest(r)) => Some(r.id),
|
|
||||||
Ok(AnyRequest::GrpcRequest(r)) => Some(r.id),
|
|
||||||
Err(_) => None,
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(Some(InternalEventPayload::WindowInfoResponse(WindowInfoResponse {
|
|
||||||
label: w.label().to_string(),
|
|
||||||
request_id,
|
|
||||||
workspace_id,
|
|
||||||
environment_id,
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
HostRequest::OtherRequest(req) => {
|
|
||||||
Ok(Some(InternalEventPayload::ErrorResponse(ErrorResponse {
|
|
||||||
error: format!(
|
|
||||||
"Unsupported plugin request in app host handler: {}",
|
|
||||||
req.type_name()
|
|
||||||
),
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,337 +0,0 @@
|
|||||||
//! Tauri-specific plugin management code.
|
|
||||||
//!
|
|
||||||
//! This module contains all Tauri integration for the plugin system:
|
|
||||||
//! - Plugin initialization and lifecycle management
|
|
||||||
//! - Tauri commands for plugin search/install/uninstall
|
|
||||||
//! - Plugin update checking
|
|
||||||
|
|
||||||
use crate::PluginContextExt;
|
|
||||||
use crate::error::Result;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use log::{error, info, warn};
|
|
||||||
use serde::Serialize;
|
|
||||||
use std::path::PathBuf;
|
|
||||||
use std::sync::Arc;
|
|
||||||
use std::sync::atomic::{AtomicBool, Ordering};
|
|
||||||
use std::time::{Duration, Instant};
|
|
||||||
use tauri::path::BaseDirectory;
|
|
||||||
use tauri::plugin::{Builder, TauriPlugin};
|
|
||||||
use tauri::{
|
|
||||||
AppHandle, Emitter, Manager, RunEvent, Runtime, State, WebviewWindow, WindowEvent, command,
|
|
||||||
is_dev,
|
|
||||||
};
|
|
||||||
use tokio::sync::Mutex;
|
|
||||||
use ts_rs::TS;
|
|
||||||
use yaak_api::yaak_api_client;
|
|
||||||
use yaak_models::models::Plugin;
|
|
||||||
use yaak_plugins::api::{
|
|
||||||
PluginNameVersion, PluginSearchResponse, PluginUpdatesResponse, check_plugin_updates,
|
|
||||||
search_plugins,
|
|
||||||
};
|
|
||||||
use yaak_plugins::events::PluginContext;
|
|
||||||
use yaak_plugins::install::{delete_and_uninstall, download_and_install};
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
use yaak_plugins::plugin_meta::get_plugin_meta;
|
|
||||||
|
|
||||||
static EXITING: AtomicBool = AtomicBool::new(false);
|
|
||||||
|
|
||||||
// ============================================================================
|
|
||||||
// Plugin Updater
|
|
||||||
// ============================================================================
|
|
||||||
|
|
||||||
const MAX_UPDATE_CHECK_HOURS: u64 = 12;
|
|
||||||
|
|
||||||
pub struct PluginUpdater {
|
|
||||||
last_check: Option<Instant>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, TS)]
|
|
||||||
#[serde(rename_all = "camelCase")]
|
|
||||||
#[ts(export, export_to = "index.ts")]
|
|
||||||
pub struct PluginUpdateNotification {
|
|
||||||
pub update_count: usize,
|
|
||||||
pub plugins: Vec<PluginUpdateInfo>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, TS)]
|
|
||||||
#[serde(rename_all = "camelCase")]
|
|
||||||
#[ts(export, export_to = "index.ts")]
|
|
||||||
pub struct PluginUpdateInfo {
|
|
||||||
pub name: String,
|
|
||||||
pub current_version: String,
|
|
||||||
pub latest_version: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl PluginUpdater {
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self { last_check: None }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn check_now<R: Runtime>(&mut self, window: &WebviewWindow<R>) -> Result<bool> {
|
|
||||||
self.last_check = Some(Instant::now());
|
|
||||||
|
|
||||||
info!("Checking for plugin updates");
|
|
||||||
|
|
||||||
let app_version = window.app_handle().package_info().version.to_string();
|
|
||||||
let http_client = yaak_api_client(&app_version)?;
|
|
||||||
let plugins = window.app_handle().db().list_plugins()?;
|
|
||||||
let updates = check_plugin_updates(&http_client, plugins.clone()).await?;
|
|
||||||
|
|
||||||
if updates.plugins.is_empty() {
|
|
||||||
info!("No plugin updates available");
|
|
||||||
return Ok(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get current plugin versions to build notification
|
|
||||||
let mut update_infos = Vec::new();
|
|
||||||
|
|
||||||
for update in &updates.plugins {
|
|
||||||
if let Some(plugin) = plugins.iter().find(|p| {
|
|
||||||
if let Ok(meta) = get_plugin_meta(&std::path::Path::new(&p.directory)) {
|
|
||||||
meta.name == update.name
|
|
||||||
} else {
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}) {
|
|
||||||
if let Ok(meta) = get_plugin_meta(&std::path::Path::new(&plugin.directory)) {
|
|
||||||
update_infos.push(PluginUpdateInfo {
|
|
||||||
name: update.name.clone(),
|
|
||||||
current_version: meta.version,
|
|
||||||
latest_version: update.version.clone(),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let notification =
|
|
||||||
PluginUpdateNotification { update_count: update_infos.len(), plugins: update_infos };
|
|
||||||
|
|
||||||
info!("Found {} plugin update(s)", notification.update_count);
|
|
||||||
|
|
||||||
if let Err(e) = window.emit_to(window.label(), "plugin_updates_available", ¬ification) {
|
|
||||||
error!("Failed to emit plugin_updates_available event: {}", e);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(true)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn maybe_check<R: Runtime>(&mut self, window: &WebviewWindow<R>) -> Result<bool> {
|
|
||||||
let update_period_seconds = MAX_UPDATE_CHECK_HOURS * 60 * 60;
|
|
||||||
|
|
||||||
if let Some(i) = self.last_check
|
|
||||||
&& i.elapsed().as_secs() < update_period_seconds
|
|
||||||
{
|
|
||||||
return Ok(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
self.check_now(window).await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ============================================================================
|
|
||||||
// Tauri Commands
|
|
||||||
// ============================================================================
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_plugins_search<R: Runtime>(
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
query: &str,
|
|
||||||
) -> Result<PluginSearchResponse> {
|
|
||||||
let app_version = app_handle.package_info().version.to_string();
|
|
||||||
let http_client = yaak_api_client(&app_version)?;
|
|
||||||
Ok(search_plugins(&http_client, query).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_plugins_install<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
name: &str,
|
|
||||||
version: Option<String>,
|
|
||||||
) -> Result<()> {
|
|
||||||
let plugin_manager = Arc::new((*window.state::<PluginManager>()).clone());
|
|
||||||
let app_version = window.app_handle().package_info().version.to_string();
|
|
||||||
let http_client = yaak_api_client(&app_version)?;
|
|
||||||
let query_manager = window.state::<yaak_models::query_manager::QueryManager>();
|
|
||||||
let plugin_context = window.plugin_context();
|
|
||||||
download_and_install(
|
|
||||||
plugin_manager,
|
|
||||||
&query_manager,
|
|
||||||
&http_client,
|
|
||||||
&plugin_context,
|
|
||||||
name,
|
|
||||||
version,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_plugins_uninstall<R: Runtime>(
|
|
||||||
plugin_id: &str,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<Plugin> {
|
|
||||||
let plugin_manager = Arc::new((*window.state::<PluginManager>()).clone());
|
|
||||||
let query_manager = window.state::<yaak_models::query_manager::QueryManager>();
|
|
||||||
let plugin_context = window.plugin_context();
|
|
||||||
Ok(delete_and_uninstall(plugin_manager, &query_manager, &plugin_context, plugin_id).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_plugins_updates<R: Runtime>(
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
) -> Result<PluginUpdatesResponse> {
|
|
||||||
let app_version = app_handle.package_info().version.to_string();
|
|
||||||
let http_client = yaak_api_client(&app_version)?;
|
|
||||||
let plugins = app_handle.db().list_plugins()?;
|
|
||||||
Ok(check_plugin_updates(&http_client, plugins).await?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_plugins_update_all<R: Runtime>(
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<Vec<PluginNameVersion>> {
|
|
||||||
let app_version = window.app_handle().package_info().version.to_string();
|
|
||||||
let http_client = yaak_api_client(&app_version)?;
|
|
||||||
let plugins = window.db().list_plugins()?;
|
|
||||||
|
|
||||||
// Get list of available updates (already filtered to only registry plugins)
|
|
||||||
let updates = check_plugin_updates(&http_client, plugins).await?;
|
|
||||||
|
|
||||||
if updates.plugins.is_empty() {
|
|
||||||
return Ok(Vec::new());
|
|
||||||
}
|
|
||||||
|
|
||||||
let plugin_manager = Arc::new((*window.state::<PluginManager>()).clone());
|
|
||||||
let query_manager = window.state::<yaak_models::query_manager::QueryManager>();
|
|
||||||
let plugin_context = window.plugin_context();
|
|
||||||
|
|
||||||
let mut updated = Vec::new();
|
|
||||||
|
|
||||||
for update in updates.plugins {
|
|
||||||
info!("Updating plugin: {} to version {}", update.name, update.version);
|
|
||||||
match download_and_install(
|
|
||||||
plugin_manager.clone(),
|
|
||||||
&query_manager,
|
|
||||||
&http_client,
|
|
||||||
&plugin_context,
|
|
||||||
&update.name,
|
|
||||||
Some(update.version.clone()),
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
Ok(_) => {
|
|
||||||
info!("Successfully updated plugin: {}", update.name);
|
|
||||||
updated.push(update.clone());
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
log::error!("Failed to update plugin {}: {:?}", update.name, e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(updated)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ============================================================================
|
|
||||||
// Tauri Plugin Initialization
|
|
||||||
// ============================================================================
|
|
||||||
|
|
||||||
pub fn init<R: Runtime>() -> TauriPlugin<R> {
|
|
||||||
Builder::new("yaak-plugins")
|
|
||||||
.setup(|app_handle, _| {
|
|
||||||
// Resolve paths for plugin manager
|
|
||||||
let vendored_plugin_dir = app_handle
|
|
||||||
.path()
|
|
||||||
.resolve("vendored/plugins", BaseDirectory::Resource)
|
|
||||||
.expect("failed to resolve plugin directory resource");
|
|
||||||
let bundled_plugin_dir = if is_dev() {
|
|
||||||
resolve_workspace_plugins_dir().unwrap_or_else(|| vendored_plugin_dir.clone())
|
|
||||||
} else {
|
|
||||||
vendored_plugin_dir.clone()
|
|
||||||
};
|
|
||||||
|
|
||||||
let installed_plugin_dir = app_handle
|
|
||||||
.path()
|
|
||||||
.app_data_dir()
|
|
||||||
.expect("failed to get app data dir")
|
|
||||||
.join("installed-plugins");
|
|
||||||
|
|
||||||
#[cfg(target_os = "windows")]
|
|
||||||
let node_bin_name = "yaaknode.exe";
|
|
||||||
#[cfg(not(target_os = "windows"))]
|
|
||||||
let node_bin_name = "yaaknode";
|
|
||||||
|
|
||||||
let node_bin_path = app_handle
|
|
||||||
.path()
|
|
||||||
.resolve(format!("vendored/node/{}", node_bin_name), BaseDirectory::Resource)
|
|
||||||
.expect("failed to resolve yaaknode binary");
|
|
||||||
|
|
||||||
let plugin_runtime_main = app_handle
|
|
||||||
.path()
|
|
||||||
.resolve("vendored/plugin-runtime", BaseDirectory::Resource)
|
|
||||||
.expect("failed to resolve plugin runtime")
|
|
||||||
.join("index.cjs");
|
|
||||||
|
|
||||||
let query_manager =
|
|
||||||
app_handle.state::<yaak_models::query_manager::QueryManager>().inner().clone();
|
|
||||||
|
|
||||||
// Create plugin manager asynchronously
|
|
||||||
let app_handle_clone = app_handle.clone();
|
|
||||||
tauri::async_runtime::block_on(async move {
|
|
||||||
let manager = PluginManager::new(
|
|
||||||
bundled_plugin_dir,
|
|
||||||
vendored_plugin_dir,
|
|
||||||
installed_plugin_dir,
|
|
||||||
node_bin_path,
|
|
||||||
plugin_runtime_main,
|
|
||||||
&query_manager,
|
|
||||||
&PluginContext::new_empty(),
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
.expect("Failed to initialize plugins");
|
|
||||||
|
|
||||||
app_handle_clone.manage(manager);
|
|
||||||
});
|
|
||||||
|
|
||||||
let plugin_updater = PluginUpdater::new();
|
|
||||||
app_handle.manage(Mutex::new(plugin_updater));
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
})
|
|
||||||
.on_event(|app, e| match e {
|
|
||||||
RunEvent::ExitRequested { api, .. } => {
|
|
||||||
if EXITING.swap(true, Ordering::SeqCst) {
|
|
||||||
return; // Only exit once to prevent infinite recursion
|
|
||||||
}
|
|
||||||
api.prevent_exit();
|
|
||||||
tauri::async_runtime::block_on(async move {
|
|
||||||
info!("Exiting plugin runtime due to app exit");
|
|
||||||
let manager: State<PluginManager> = app.state();
|
|
||||||
manager.terminate().await;
|
|
||||||
app.exit(0);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
RunEvent::WindowEvent { event: WindowEvent::Focused(true), label, .. } => {
|
|
||||||
// Check for plugin updates on window focus
|
|
||||||
let w = app.get_webview_window(&label).unwrap();
|
|
||||||
let h = app.clone();
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
tokio::time::sleep(Duration::from_secs(3)).await;
|
|
||||||
let val: State<'_, Mutex<PluginUpdater>> = h.state();
|
|
||||||
if let Err(e) = val.lock().await.maybe_check(&w).await {
|
|
||||||
warn!("Failed to check for plugin updates {e:?}");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
})
|
|
||||||
.build()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn resolve_workspace_plugins_dir() -> Option<PathBuf> {
|
|
||||||
PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
|
||||||
.join("../..")
|
|
||||||
.join("plugins")
|
|
||||||
.canonicalize()
|
|
||||||
.ok()
|
|
||||||
}
|
|
||||||
@@ -1,85 +0,0 @@
|
|||||||
use log::info;
|
|
||||||
use serde_json::Value;
|
|
||||||
use std::collections::BTreeMap;
|
|
||||||
pub use yaak::render::render_http_request;
|
|
||||||
use yaak_models::models::{Environment, GrpcRequest, HttpRequestHeader};
|
|
||||||
use yaak_models::render::make_vars_hashmap;
|
|
||||||
use yaak_templates::{RenderOptions, TemplateCallback, parse_and_render, render_json_value_raw};
|
|
||||||
|
|
||||||
pub async fn render_template<T: TemplateCallback>(
|
|
||||||
template: &str,
|
|
||||||
environment_chain: Vec<Environment>,
|
|
||||||
cb: &T,
|
|
||||||
opt: &RenderOptions,
|
|
||||||
) -> yaak_templates::error::Result<String> {
|
|
||||||
let vars = &make_vars_hashmap(environment_chain);
|
|
||||||
parse_and_render(template, vars, cb, &opt).await
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn render_json_value<T: TemplateCallback>(
|
|
||||||
value: Value,
|
|
||||||
environment_chain: Vec<Environment>,
|
|
||||||
cb: &T,
|
|
||||||
opt: &RenderOptions,
|
|
||||||
) -> yaak_templates::error::Result<Value> {
|
|
||||||
let vars = &make_vars_hashmap(environment_chain);
|
|
||||||
render_json_value_raw(value, vars, cb, opt).await
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn render_grpc_request<T: TemplateCallback>(
|
|
||||||
r: &GrpcRequest,
|
|
||||||
environment_chain: Vec<Environment>,
|
|
||||||
cb: &T,
|
|
||||||
opt: &RenderOptions,
|
|
||||||
) -> yaak_templates::error::Result<GrpcRequest> {
|
|
||||||
let vars = &make_vars_hashmap(environment_chain);
|
|
||||||
|
|
||||||
let mut metadata = Vec::new();
|
|
||||||
for p in r.metadata.clone() {
|
|
||||||
if !p.enabled {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
metadata.push(HttpRequestHeader {
|
|
||||||
enabled: p.enabled,
|
|
||||||
name: parse_and_render(p.name.as_str(), vars, cb, &opt).await?,
|
|
||||||
value: parse_and_render(p.value.as_str(), vars, cb, &opt).await?,
|
|
||||||
id: p.id,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
let authentication = {
|
|
||||||
let mut disabled = false;
|
|
||||||
let mut auth = BTreeMap::new();
|
|
||||||
match r.authentication.get("disabled") {
|
|
||||||
Some(Value::Bool(true)) => {
|
|
||||||
disabled = true;
|
|
||||||
}
|
|
||||||
Some(Value::String(tmpl)) => {
|
|
||||||
disabled = parse_and_render(tmpl.as_str(), vars, cb, &opt)
|
|
||||||
.await
|
|
||||||
.unwrap_or_default()
|
|
||||||
.is_empty();
|
|
||||||
info!(
|
|
||||||
"Rendering authentication.disabled as a template: {disabled} from \"{tmpl}\""
|
|
||||||
);
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
if disabled {
|
|
||||||
auth.insert("disabled".to_string(), Value::Bool(true));
|
|
||||||
} else {
|
|
||||||
for (k, v) in r.authentication.clone() {
|
|
||||||
if k == "disabled" {
|
|
||||||
auth.insert(k, Value::Bool(false));
|
|
||||||
} else {
|
|
||||||
auth.insert(k, render_json_value_raw(v, vars, cb, &opt).await?);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
auth
|
|
||||||
};
|
|
||||||
|
|
||||||
let url = parse_and_render(r.url.as_str(), vars, cb, &opt).await?;
|
|
||||||
|
|
||||||
Ok(GrpcRequest { url, metadata, authentication, ..r.to_owned() })
|
|
||||||
}
|
|
||||||
@@ -1,405 +0,0 @@
|
|||||||
use std::fmt::{Display, Formatter};
|
|
||||||
use std::path::PathBuf;
|
|
||||||
use std::time::{Duration, Instant};
|
|
||||||
|
|
||||||
use crate::error::Result;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use log::{debug, error, info, warn};
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
use tauri::{Emitter, Listener, Manager, Runtime, WebviewWindow};
|
|
||||||
use tauri_plugin_dialog::{DialogExt, MessageDialogButtons};
|
|
||||||
use tauri_plugin_updater::{Update, UpdaterExt};
|
|
||||||
use tokio::task::block_in_place;
|
|
||||||
use tokio::time::sleep;
|
|
||||||
use ts_rs::TS;
|
|
||||||
use yaak_models::util::generate_id;
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
|
|
||||||
use url::Url;
|
|
||||||
use yaak_api::get_system_proxy_url;
|
|
||||||
|
|
||||||
use crate::error::Error::GenericError;
|
|
||||||
use crate::is_dev;
|
|
||||||
|
|
||||||
const MAX_UPDATE_CHECK_HOURS_STABLE: u64 = 12;
|
|
||||||
const MAX_UPDATE_CHECK_HOURS_BETA: u64 = 3;
|
|
||||||
const MAX_UPDATE_CHECK_HOURS_ALPHA: u64 = 1;
|
|
||||||
|
|
||||||
// Create updater struct
|
|
||||||
pub struct YaakUpdater {
|
|
||||||
last_check: Option<Instant>,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub enum UpdateMode {
|
|
||||||
Stable,
|
|
||||||
Beta,
|
|
||||||
Alpha,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Display for UpdateMode {
|
|
||||||
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
|
||||||
let s = match self {
|
|
||||||
UpdateMode::Stable => "stable",
|
|
||||||
UpdateMode::Beta => "beta",
|
|
||||||
UpdateMode::Alpha => "alpha",
|
|
||||||
};
|
|
||||||
write!(f, "{}", s)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl UpdateMode {
|
|
||||||
pub fn new(mode: &str) -> UpdateMode {
|
|
||||||
match mode {
|
|
||||||
"beta" => UpdateMode::Beta,
|
|
||||||
"alpha" => UpdateMode::Alpha,
|
|
||||||
_ => UpdateMode::Stable,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(PartialEq)]
|
|
||||||
pub enum UpdateTrigger {
|
|
||||||
Background,
|
|
||||||
User,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl YaakUpdater {
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self { last_check: None }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn check_now<R: Runtime>(
|
|
||||||
&mut self,
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
mode: UpdateMode,
|
|
||||||
auto_download: bool,
|
|
||||||
update_trigger: UpdateTrigger,
|
|
||||||
) -> Result<bool> {
|
|
||||||
// Only AppImage supports updates on Linux, so skip if it's not
|
|
||||||
#[cfg(target_os = "linux")]
|
|
||||||
{
|
|
||||||
if std::env::var("APPIMAGE").is_err() {
|
|
||||||
return Ok(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let settings = window.db().get_settings();
|
|
||||||
let update_key = format!("{:x}", md5::compute(settings.id));
|
|
||||||
self.last_check = Some(Instant::now());
|
|
||||||
|
|
||||||
info!("Checking for updates mode={} autodl={}", mode, auto_download);
|
|
||||||
|
|
||||||
let w = window.clone();
|
|
||||||
let mut updater_builder = w.updater_builder();
|
|
||||||
if let Some(proxy_url) = get_system_proxy_url() {
|
|
||||||
if let Ok(url) = Url::parse(&proxy_url) {
|
|
||||||
updater_builder = updater_builder.proxy(url);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let update_check_result = updater_builder
|
|
||||||
.on_before_exit(move || {
|
|
||||||
// Kill plugin manager before exit or NSIS installer will fail to replace sidecar
|
|
||||||
// while it's running.
|
|
||||||
// NOTE: This is only called on Windows
|
|
||||||
let w = w.clone();
|
|
||||||
block_in_place(|| {
|
|
||||||
tauri::async_runtime::block_on(async move {
|
|
||||||
info!("Shutting down plugin manager before update");
|
|
||||||
let plugin_manager = w.state::<PluginManager>();
|
|
||||||
plugin_manager.terminate().await;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.header("X-Update-Mode", mode.to_string())?
|
|
||||||
.header("X-Update-Key", update_key)?
|
|
||||||
.header(
|
|
||||||
"X-Update-Trigger",
|
|
||||||
match update_trigger {
|
|
||||||
UpdateTrigger::Background => "background",
|
|
||||||
UpdateTrigger::User => "user",
|
|
||||||
},
|
|
||||||
)?
|
|
||||||
.header("X-Install-Mode", detect_install_mode().unwrap_or("unknown"))?
|
|
||||||
.build()?
|
|
||||||
.check()
|
|
||||||
.await;
|
|
||||||
|
|
||||||
let result = match update_check_result? {
|
|
||||||
None => false,
|
|
||||||
Some(update) => {
|
|
||||||
let w = window.clone();
|
|
||||||
tauri::async_runtime::spawn(async move {
|
|
||||||
// Force native updater if specified (useful if a release broke the UI)
|
|
||||||
let native_install_mode =
|
|
||||||
update.raw_json.get("install_mode").map(|v| v.as_str()).unwrap_or_default()
|
|
||||||
== Some("native");
|
|
||||||
if native_install_mode {
|
|
||||||
start_native_update(&w, &update).await;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If it's a background update, try downloading it first
|
|
||||||
if update_trigger == UpdateTrigger::Background && auto_download {
|
|
||||||
info!("Downloading update {} in background", update.version);
|
|
||||||
if let Err(e) = download_update_idempotent(&w, &update).await {
|
|
||||||
error!("Failed to download {}: {}", update.version, e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
match start_integrated_update(&w, &update).await {
|
|
||||||
Ok(UpdateResponseAction::Skip) => {
|
|
||||||
info!("Confirmed {}: skipped", update.version);
|
|
||||||
}
|
|
||||||
Ok(UpdateResponseAction::Install) => {
|
|
||||||
info!("Confirmed {}: install", update.version);
|
|
||||||
if let Err(e) = install_update_maybe_download(&w, &update).await {
|
|
||||||
error!("Failed to install: {e}");
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
|
|
||||||
info!("Installed {}", update.version);
|
|
||||||
finish_integrated_update(&w, &update).await;
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
warn!("Failed to notify frontend, falling back: {e}",);
|
|
||||||
start_native_update(&w, &update).await;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
});
|
|
||||||
true
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(result)
|
|
||||||
}
|
|
||||||
pub async fn maybe_check<R: Runtime>(
|
|
||||||
&mut self,
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
auto_download: bool,
|
|
||||||
mode: UpdateMode,
|
|
||||||
) -> Result<bool> {
|
|
||||||
let update_period_seconds = match mode {
|
|
||||||
UpdateMode::Stable => MAX_UPDATE_CHECK_HOURS_STABLE,
|
|
||||||
UpdateMode::Beta => MAX_UPDATE_CHECK_HOURS_BETA,
|
|
||||||
UpdateMode::Alpha => MAX_UPDATE_CHECK_HOURS_ALPHA,
|
|
||||||
} * (60 * 60);
|
|
||||||
|
|
||||||
if let Some(i) = self.last_check
|
|
||||||
&& i.elapsed().as_secs() < update_period_seconds
|
|
||||||
{
|
|
||||||
return Ok(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't check if development (can still with manual user trigger)
|
|
||||||
if is_dev() {
|
|
||||||
return Ok(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
self.check_now(window, mode, auto_download, UpdateTrigger::Background).await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Default, TS)]
|
|
||||||
#[serde(default, rename_all = "camelCase")]
|
|
||||||
#[ts(export, export_to = "index.ts")]
|
|
||||||
struct UpdateInfo {
|
|
||||||
reply_event_id: String,
|
|
||||||
version: String,
|
|
||||||
downloaded: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Deserialize, TS)]
|
|
||||||
#[serde(rename_all = "camelCase", tag = "type")]
|
|
||||||
#[ts(export, export_to = "index.ts")]
|
|
||||||
enum UpdateResponse {
|
|
||||||
Ack,
|
|
||||||
Action { action: UpdateResponseAction },
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Deserialize, TS)]
|
|
||||||
#[serde(rename_all = "snake_case")]
|
|
||||||
#[ts(export, export_to = "index.ts")]
|
|
||||||
enum UpdateResponseAction {
|
|
||||||
Install,
|
|
||||||
Skip,
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn finish_integrated_update<R: Runtime>(window: &WebviewWindow<R>, update: &Update) {
|
|
||||||
if let Err(e) = window.emit_to(window.label(), "update_installed", update.version.to_string()) {
|
|
||||||
warn!("Failed to notify frontend of update install: {}", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn start_integrated_update<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
update: &Update,
|
|
||||||
) -> Result<UpdateResponseAction> {
|
|
||||||
let download_path = ensure_download_path(window, update)?;
|
|
||||||
debug!("Download path: {}", download_path.display());
|
|
||||||
let downloaded = download_path.exists();
|
|
||||||
let ack_wait = Duration::from_secs(3);
|
|
||||||
let reply_id = generate_id();
|
|
||||||
|
|
||||||
// 1) Start listening BEFORE emitting to avoid missing a fast reply
|
|
||||||
let (tx, mut rx) = tokio::sync::mpsc::unbounded_channel::<UpdateResponse>();
|
|
||||||
let w_for_listener = window.clone();
|
|
||||||
|
|
||||||
let event_id = w_for_listener.listen(reply_id.clone(), move |ev| {
|
|
||||||
match serde_json::from_str::<UpdateResponse>(ev.payload()) {
|
|
||||||
Ok(UpdateResponse::Ack) => {
|
|
||||||
let _ = tx.send(UpdateResponse::Ack);
|
|
||||||
}
|
|
||||||
Ok(UpdateResponse::Action { action }) => {
|
|
||||||
let _ = tx.send(UpdateResponse::Action { action });
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
warn!("Failed to parse update reply from frontend: {e:?}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Make sure we always unlisten
|
|
||||||
struct Unlisten<'a, R: Runtime> {
|
|
||||||
win: &'a WebviewWindow<R>,
|
|
||||||
id: tauri::EventId,
|
|
||||||
}
|
|
||||||
impl<'a, R: Runtime> Drop for Unlisten<'a, R> {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
self.win.unlisten(self.id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let _guard = Unlisten { win: window, id: event_id };
|
|
||||||
|
|
||||||
// 2) Emit the event now that listener is in place
|
|
||||||
let info =
|
|
||||||
UpdateInfo { version: update.version.to_string(), downloaded, reply_event_id: reply_id };
|
|
||||||
window
|
|
||||||
.emit_to(window.label(), "update_available", &info)
|
|
||||||
.map_err(|e| GenericError(format!("Failed to emit update_available: {e}")))?;
|
|
||||||
|
|
||||||
// 3) Two-stage timeout: first wait for ack, then wait for final action
|
|
||||||
// --- Phase 1: wait for ACK with timeout ---
|
|
||||||
let ack_timer = sleep(ack_wait);
|
|
||||||
tokio::pin!(ack_timer);
|
|
||||||
|
|
||||||
loop {
|
|
||||||
tokio::select! {
|
|
||||||
msg = rx.recv() => match msg {
|
|
||||||
Some(UpdateResponse::Ack) => break, // proceed to Phase 2
|
|
||||||
Some(UpdateResponse::Action{action}) => return Ok(action), // user was fast
|
|
||||||
None => return Err(GenericError("frontend channel closed before ack".into())),
|
|
||||||
},
|
|
||||||
_ = &mut ack_timer => {
|
|
||||||
return Err(GenericError("timed out waiting for frontend ack".into()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// --- Phase 2: wait forever for final action ---
|
|
||||||
loop {
|
|
||||||
match rx.recv().await {
|
|
||||||
Some(UpdateResponse::Action { action }) => return Ok(action),
|
|
||||||
Some(UpdateResponse::Ack) => { /* ignore extra acks */ }
|
|
||||||
None => return Err(GenericError("frontend channel closed before action".into())),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn start_native_update<R: Runtime>(window: &WebviewWindow<R>, update: &Update) {
|
|
||||||
// If the frontend doesn't respond, fallback to native dialogs
|
|
||||||
let confirmed = window
|
|
||||||
.dialog()
|
|
||||||
.message(format!(
|
|
||||||
"{} is available. Would you like to download and install it now?",
|
|
||||||
update.version
|
|
||||||
))
|
|
||||||
.buttons(MessageDialogButtons::OkCancelCustom("Download".to_string(), "Later".to_string()))
|
|
||||||
.title("Update Available")
|
|
||||||
.blocking_show();
|
|
||||||
if !confirmed {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
match update.download_and_install(|_, _| {}, || {}).await {
|
|
||||||
Ok(()) => {
|
|
||||||
if window
|
|
||||||
.dialog()
|
|
||||||
.message("Would you like to restart the app?")
|
|
||||||
.title("Update Installed")
|
|
||||||
.buttons(MessageDialogButtons::OkCancelCustom(
|
|
||||||
"Restart".to_string(),
|
|
||||||
"Later".to_string(),
|
|
||||||
))
|
|
||||||
.blocking_show()
|
|
||||||
{
|
|
||||||
window.app_handle().request_restart();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
window.dialog().message(format!("The update failed to install: {}", e));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn download_update_idempotent<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
update: &Update,
|
|
||||||
) -> Result<PathBuf> {
|
|
||||||
let dl_path = ensure_download_path(window, update)?;
|
|
||||||
|
|
||||||
if dl_path.exists() {
|
|
||||||
info!("{} already downloaded to {}", update.version, dl_path.display());
|
|
||||||
return Ok(dl_path);
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("{} downloading: {}", update.version, dl_path.display());
|
|
||||||
let dl_bytes = update.download(|_, _| {}, || {}).await?;
|
|
||||||
std::fs::write(&dl_path, dl_bytes)
|
|
||||||
.map_err(|e| GenericError(format!("Failed to write update: {e}")))?;
|
|
||||||
|
|
||||||
info!("{} downloaded", update.version);
|
|
||||||
|
|
||||||
Ok(dl_path)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Detect the installer type so the update server can serve the correct artifact.
|
|
||||||
fn detect_install_mode() -> Option<&'static str> {
|
|
||||||
#[cfg(target_os = "windows")]
|
|
||||||
{
|
|
||||||
if let Ok(exe) = std::env::current_exe() {
|
|
||||||
let path = exe.to_string_lossy().to_lowercase();
|
|
||||||
if path.starts_with(r"c:\program files") {
|
|
||||||
return Some("nsis-machine");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Some("nsis");
|
|
||||||
}
|
|
||||||
#[allow(unreachable_code)]
|
|
||||||
None
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn install_update_maybe_download<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
update: &Update,
|
|
||||||
) -> Result<()> {
|
|
||||||
let dl_path = download_update_idempotent(window, update).await?;
|
|
||||||
let update_bytes = std::fs::read(&dl_path)?;
|
|
||||||
update.install(update_bytes.as_slice())?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn ensure_download_path<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
update: &Update,
|
|
||||||
) -> Result<PathBuf> {
|
|
||||||
// Ensure dir exists
|
|
||||||
let base_dir = window.path().app_cache_dir()?.join("updates");
|
|
||||||
std::fs::create_dir_all(&base_dir)?;
|
|
||||||
|
|
||||||
// Generate name based on signature
|
|
||||||
let sig_digest = md5::compute(&update.signature);
|
|
||||||
let name = format!("yaak-{}-{:x}", update.version, sig_digest);
|
|
||||||
let dl_path = base_dir.join(name);
|
|
||||||
|
|
||||||
Ok(dl_path)
|
|
||||||
}
|
|
||||||
@@ -1,136 +0,0 @@
|
|||||||
use crate::PluginContextExt;
|
|
||||||
use crate::error::Result;
|
|
||||||
use crate::import::import_data;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use log::{info, warn};
|
|
||||||
use std::collections::HashMap;
|
|
||||||
use std::fs;
|
|
||||||
use std::sync::Arc;
|
|
||||||
use tauri::{AppHandle, Emitter, Manager, Runtime, Url};
|
|
||||||
use tauri_plugin_dialog::{DialogExt, MessageDialogButtons, MessageDialogKind};
|
|
||||||
use yaak_api::yaak_api_client;
|
|
||||||
use yaak_models::util::generate_id;
|
|
||||||
use yaak_plugins::events::{Color, ShowToastRequest};
|
|
||||||
use yaak_plugins::install::download_and_install;
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
|
|
||||||
pub(crate) async fn handle_deep_link<R: Runtime>(
|
|
||||||
app_handle: &AppHandle<R>,
|
|
||||||
url: &Url,
|
|
||||||
) -> Result<()> {
|
|
||||||
let command = url.domain().unwrap_or_default();
|
|
||||||
info!("Yaak URI scheme invoked {}?{}", command, url.query().unwrap_or_default());
|
|
||||||
|
|
||||||
let query_map: HashMap<String, String> = url.query_pairs().into_owned().collect();
|
|
||||||
let windows = app_handle.webview_windows();
|
|
||||||
let (_, window) = windows.iter().next().unwrap();
|
|
||||||
|
|
||||||
match command {
|
|
||||||
"install-plugin" => {
|
|
||||||
let name = query_map.get("name").unwrap();
|
|
||||||
let version = query_map.get("version").cloned();
|
|
||||||
_ = window.set_focus();
|
|
||||||
let confirmed_install = app_handle
|
|
||||||
.dialog()
|
|
||||||
.message(format!("Install plugin {name} {version:?}?"))
|
|
||||||
.kind(MessageDialogKind::Info)
|
|
||||||
.buttons(MessageDialogButtons::OkCancelCustom(
|
|
||||||
"Install".to_string(),
|
|
||||||
"Cancel".to_string(),
|
|
||||||
))
|
|
||||||
.blocking_show();
|
|
||||||
if !confirmed_install {
|
|
||||||
// Cancelled installation
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let plugin_manager = Arc::new((*window.state::<PluginManager>()).clone());
|
|
||||||
let query_manager = app_handle.db_manager();
|
|
||||||
let app_version = app_handle.package_info().version.to_string();
|
|
||||||
let http_client = yaak_api_client(&app_version)?;
|
|
||||||
let plugin_context = window.plugin_context();
|
|
||||||
let pv = download_and_install(
|
|
||||||
plugin_manager,
|
|
||||||
&query_manager,
|
|
||||||
&http_client,
|
|
||||||
&plugin_context,
|
|
||||||
name,
|
|
||||||
version,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
app_handle.emit(
|
|
||||||
"show_toast",
|
|
||||||
ShowToastRequest {
|
|
||||||
message: format!("Installed {name}@{}", pv.version),
|
|
||||||
color: Some(Color::Success),
|
|
||||||
icon: None,
|
|
||||||
timeout: Some(5000),
|
|
||||||
},
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
"import-data" => {
|
|
||||||
let mut file_path = query_map.get("path").map(|s| s.to_owned());
|
|
||||||
let name = query_map.get("name").map(|s| s.to_owned()).unwrap_or("data".to_string());
|
|
||||||
_ = window.set_focus();
|
|
||||||
|
|
||||||
if let Some(file_url) = query_map.get("url") {
|
|
||||||
let confirmed_import = app_handle
|
|
||||||
.dialog()
|
|
||||||
.message(format!("Import {name} from {file_url}?"))
|
|
||||||
.kind(MessageDialogKind::Info)
|
|
||||||
.buttons(MessageDialogButtons::OkCancelCustom(
|
|
||||||
"Import".to_string(),
|
|
||||||
"Cancel".to_string(),
|
|
||||||
))
|
|
||||||
.blocking_show();
|
|
||||||
if !confirmed_import {
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let app_version = app_handle.package_info().version.to_string();
|
|
||||||
let resp = yaak_api_client(&app_version)?.get(file_url).send().await?;
|
|
||||||
let json = resp.bytes().await?;
|
|
||||||
let p = app_handle
|
|
||||||
.path()
|
|
||||||
.temp_dir()?
|
|
||||||
.join(format!("import-{}", generate_id()))
|
|
||||||
.to_string_lossy()
|
|
||||||
.to_string();
|
|
||||||
fs::write(&p, json)?;
|
|
||||||
file_path = Some(p);
|
|
||||||
}
|
|
||||||
|
|
||||||
let file_path = match file_path {
|
|
||||||
Some(p) => p,
|
|
||||||
None => {
|
|
||||||
app_handle.emit(
|
|
||||||
"show_toast",
|
|
||||||
ShowToastRequest {
|
|
||||||
message: "Failed to import data".to_string(),
|
|
||||||
color: Some(Color::Danger),
|
|
||||||
icon: None,
|
|
||||||
timeout: None,
|
|
||||||
},
|
|
||||||
)?;
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let results = import_data(window, &file_path).await?;
|
|
||||||
window.emit(
|
|
||||||
"show_toast",
|
|
||||||
ShowToastRequest {
|
|
||||||
message: format!("Imported data for {} workspaces", results.workspaces.len()),
|
|
||||||
color: Some(Color::Success),
|
|
||||||
icon: None,
|
|
||||||
timeout: Some(5000),
|
|
||||||
},
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
warn!("Unknown deep link command: {command}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
@@ -1,447 +0,0 @@
|
|||||||
//! WebSocket Tauri command wrappers
|
|
||||||
//! These wrap the core yaak-ws functionality for Tauri IPC.
|
|
||||||
|
|
||||||
use crate::PluginContextExt;
|
|
||||||
use crate::error::Result;
|
|
||||||
use crate::models_ext::QueryManagerExt;
|
|
||||||
use http::HeaderMap;
|
|
||||||
use log::{debug, info, warn};
|
|
||||||
use std::str::FromStr;
|
|
||||||
use std::sync::Arc;
|
|
||||||
use tauri::http::HeaderValue;
|
|
||||||
use tauri::{AppHandle, Manager, Runtime, State, WebviewWindow, command};
|
|
||||||
use tokio::sync::{Mutex, mpsc};
|
|
||||||
use tokio_tungstenite::tungstenite::Message;
|
|
||||||
use url::Url;
|
|
||||||
use yaak_crypto::manager::EncryptionManager;
|
|
||||||
use yaak_http::cookies::CookieStore;
|
|
||||||
use yaak_http::path_placeholders::apply_path_placeholders;
|
|
||||||
use yaak_models::models::{
|
|
||||||
HttpResponseHeader, WebsocketConnection, WebsocketConnectionState, WebsocketEvent,
|
|
||||||
WebsocketEventType, WebsocketRequest,
|
|
||||||
};
|
|
||||||
use yaak_models::util::UpdateSource;
|
|
||||||
use yaak_plugins::events::{CallHttpAuthenticationRequest, HttpHeader, RenderPurpose};
|
|
||||||
use yaak_plugins::manager::PluginManager;
|
|
||||||
use yaak_plugins::template_callback::PluginTemplateCallback;
|
|
||||||
use yaak_templates::{RenderErrorBehavior, RenderOptions};
|
|
||||||
use yaak_tls::find_client_certificate;
|
|
||||||
use yaak_ws::{WebsocketManager, render_websocket_request};
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_delete_connections<R: Runtime>(
|
|
||||||
request_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
) -> Result<()> {
|
|
||||||
Ok(app_handle.db().delete_all_websocket_connections_for_request(
|
|
||||||
request_id,
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_send<R: Runtime>(
|
|
||||||
connection_id: &str,
|
|
||||||
environment_id: Option<&str>,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
ws_manager: State<'_, Mutex<WebsocketManager>>,
|
|
||||||
) -> Result<WebsocketConnection> {
|
|
||||||
let connection = app_handle.db().get_websocket_connection(connection_id)?;
|
|
||||||
let unrendered_request = app_handle.db().get_websocket_request(&connection.request_id)?;
|
|
||||||
let environment_chain = app_handle.db().resolve_environments(
|
|
||||||
&unrendered_request.workspace_id,
|
|
||||||
unrendered_request.folder_id.as_deref(),
|
|
||||||
environment_id,
|
|
||||||
)?;
|
|
||||||
let (resolved_request, _auth_context_id) =
|
|
||||||
resolve_websocket_request(&window, &unrendered_request)?;
|
|
||||||
let plugin_manager = Arc::new((*app_handle.state::<PluginManager>()).clone());
|
|
||||||
let encryption_manager = Arc::new((*app_handle.state::<EncryptionManager>()).clone());
|
|
||||||
let request = render_websocket_request(
|
|
||||||
&resolved_request,
|
|
||||||
environment_chain,
|
|
||||||
&PluginTemplateCallback::new(
|
|
||||||
plugin_manager,
|
|
||||||
encryption_manager,
|
|
||||||
&window.plugin_context(),
|
|
||||||
RenderPurpose::Send,
|
|
||||||
),
|
|
||||||
&RenderOptions { error_behavior: RenderErrorBehavior::Throw },
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let mut ws_manager = ws_manager.lock().await;
|
|
||||||
ws_manager.send(&connection.id, Message::Text(request.message.clone().into())).await?;
|
|
||||||
|
|
||||||
app_handle.db().upsert_websocket_event(
|
|
||||||
&WebsocketEvent {
|
|
||||||
connection_id: connection.id.clone(),
|
|
||||||
request_id: request.id.clone(),
|
|
||||||
workspace_id: connection.workspace_id.clone(),
|
|
||||||
is_server: false,
|
|
||||||
message_type: WebsocketEventType::Text,
|
|
||||||
message: request.message.into(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
Ok(connection)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_close<R: Runtime>(
|
|
||||||
connection_id: &str,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
ws_manager: State<'_, Mutex<WebsocketManager>>,
|
|
||||||
) -> Result<WebsocketConnection> {
|
|
||||||
let connection = {
|
|
||||||
let db = app_handle.db();
|
|
||||||
let connection = db.get_websocket_connection(connection_id)?;
|
|
||||||
db.upsert_websocket_connection(
|
|
||||||
&WebsocketConnection { state: WebsocketConnectionState::Closing, ..connection },
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut ws_manager = ws_manager.lock().await;
|
|
||||||
if let Err(e) = ws_manager.close(&connection.id).await {
|
|
||||||
warn!("Failed to close WebSocket connection: {e:?}");
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(connection)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub async fn cmd_ws_connect<R: Runtime>(
|
|
||||||
request_id: &str,
|
|
||||||
environment_id: Option<&str>,
|
|
||||||
cookie_jar_id: Option<&str>,
|
|
||||||
app_handle: AppHandle<R>,
|
|
||||||
window: WebviewWindow<R>,
|
|
||||||
_plugin_manager: State<'_, PluginManager>,
|
|
||||||
ws_manager: State<'_, Mutex<WebsocketManager>>,
|
|
||||||
) -> Result<WebsocketConnection> {
|
|
||||||
let unrendered_request = app_handle.db().get_websocket_request(request_id)?;
|
|
||||||
let environment_chain = app_handle.db().resolve_environments(
|
|
||||||
&unrendered_request.workspace_id,
|
|
||||||
unrendered_request.folder_id.as_deref(),
|
|
||||||
environment_id,
|
|
||||||
)?;
|
|
||||||
let workspace = app_handle.db().get_workspace(&unrendered_request.workspace_id)?;
|
|
||||||
let settings = app_handle.db().get_settings();
|
|
||||||
let (resolved_request, auth_context_id) =
|
|
||||||
resolve_websocket_request(&window, &unrendered_request)?;
|
|
||||||
let plugin_manager = Arc::new((*app_handle.state::<PluginManager>()).clone());
|
|
||||||
let encryption_manager = Arc::new((*app_handle.state::<EncryptionManager>()).clone());
|
|
||||||
let request = render_websocket_request(
|
|
||||||
&resolved_request,
|
|
||||||
environment_chain,
|
|
||||||
&PluginTemplateCallback::new(
|
|
||||||
plugin_manager.clone(),
|
|
||||||
encryption_manager.clone(),
|
|
||||||
&window.plugin_context(),
|
|
||||||
RenderPurpose::Send,
|
|
||||||
),
|
|
||||||
&RenderOptions { error_behavior: RenderErrorBehavior::Throw },
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let connection = app_handle.db().upsert_websocket_connection(
|
|
||||||
&WebsocketConnection {
|
|
||||||
workspace_id: request.workspace_id.clone(),
|
|
||||||
request_id: request_id.to_string(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let (mut url, url_parameters) = apply_path_placeholders(&request.url, &request.url_parameters);
|
|
||||||
if !url.starts_with("ws://") && !url.starts_with("wss://") {
|
|
||||||
url.insert_str(0, "ws://");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add URL parameters to URL
|
|
||||||
let mut url = match Url::parse(&url) {
|
|
||||||
Ok(url) => url,
|
|
||||||
Err(e) => {
|
|
||||||
return Ok(app_handle.db().upsert_websocket_connection(
|
|
||||||
&WebsocketConnection {
|
|
||||||
error: Some(format!("Failed to parse URL {}", e.to_string())),
|
|
||||||
state: WebsocketConnectionState::Closed,
|
|
||||||
..connection
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut headers = HeaderMap::new();
|
|
||||||
|
|
||||||
for h in request.headers.clone() {
|
|
||||||
if h.name.is_empty() && h.value.is_empty() {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if !h.enabled {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
headers.insert(
|
|
||||||
http::HeaderName::from_str(&h.name).unwrap(),
|
|
||||||
HeaderValue::from_str(&h.value).unwrap(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
match request.authentication_type {
|
|
||||||
None => {
|
|
||||||
// No authentication found. Not even inherited
|
|
||||||
}
|
|
||||||
Some(authentication_type) if authentication_type == "none" => {
|
|
||||||
// Explicitly no authentication
|
|
||||||
}
|
|
||||||
Some(authentication_type) => {
|
|
||||||
let auth = request.authentication.clone();
|
|
||||||
let plugin_req = CallHttpAuthenticationRequest {
|
|
||||||
context_id: format!("{:x}", md5::compute(auth_context_id)),
|
|
||||||
values: serde_json::from_value(serde_json::to_value(&auth).unwrap()).unwrap(),
|
|
||||||
method: "POST".to_string(),
|
|
||||||
url: request.url.clone(),
|
|
||||||
headers: request
|
|
||||||
.headers
|
|
||||||
.clone()
|
|
||||||
.into_iter()
|
|
||||||
.map(|h| HttpHeader { name: h.name, value: h.value })
|
|
||||||
.collect(),
|
|
||||||
};
|
|
||||||
let plugin_result = plugin_manager
|
|
||||||
.call_http_authentication(
|
|
||||||
&window.plugin_context(),
|
|
||||||
&authentication_type,
|
|
||||||
plugin_req,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
for header in plugin_result.set_headers.unwrap_or_default() {
|
|
||||||
match (
|
|
||||||
http::HeaderName::from_str(&header.name),
|
|
||||||
HeaderValue::from_str(&header.value),
|
|
||||||
) {
|
|
||||||
(Ok(name), Ok(value)) => {
|
|
||||||
headers.insert(name, value);
|
|
||||||
}
|
|
||||||
_ => continue,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
if let Some(params) = plugin_result.set_query_parameters {
|
|
||||||
let mut query_pairs = url.query_pairs_mut();
|
|
||||||
for p in params {
|
|
||||||
query_pairs.append_pair(&p.name, &p.value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add cookies to WS HTTP Upgrade
|
|
||||||
if let Some(id) = cookie_jar_id {
|
|
||||||
let cookie_jar = app_handle.db().get_cookie_jar(&id)?;
|
|
||||||
let store = CookieStore::from_cookies(cookie_jar.cookies);
|
|
||||||
|
|
||||||
// Convert WS URL -> HTTP URL because our cookie store matches based on
|
|
||||||
// Path/HttpOnly/Secure attributes even though WS upgrades are HTTP requests
|
|
||||||
let http_url = convert_ws_url_to_http(&url);
|
|
||||||
if let Some(cookie_header_value) = store.get_cookie_header(&http_url) {
|
|
||||||
debug!("Inserting cookies into WS upgrade to {}: {}", url, cookie_header_value);
|
|
||||||
headers.insert(
|
|
||||||
http::HeaderName::from_static("cookie"),
|
|
||||||
HeaderValue::from_str(&cookie_header_value).unwrap(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let (receive_tx, mut receive_rx) = mpsc::channel::<Message>(128);
|
|
||||||
let mut ws_manager = ws_manager.lock().await;
|
|
||||||
|
|
||||||
{
|
|
||||||
let valid_query_pairs = url_parameters
|
|
||||||
.into_iter()
|
|
||||||
.filter(|p| p.enabled && !p.name.is_empty())
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
// NOTE: Only mutate query pairs if there are any, or it will append an empty `?` to the URL
|
|
||||||
if !valid_query_pairs.is_empty() {
|
|
||||||
let mut query_pairs = url.query_pairs_mut();
|
|
||||||
for p in valid_query_pairs {
|
|
||||||
query_pairs.append_pair(p.name.as_str(), p.value.as_str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let client_cert = find_client_certificate(url.as_str(), &settings.client_certificates);
|
|
||||||
|
|
||||||
let response = match ws_manager
|
|
||||||
.connect(
|
|
||||||
&connection.id,
|
|
||||||
url.as_str(),
|
|
||||||
headers,
|
|
||||||
receive_tx,
|
|
||||||
workspace.setting_validate_certificates,
|
|
||||||
client_cert,
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
Ok(r) => r,
|
|
||||||
Err(e) => {
|
|
||||||
return Ok(app_handle.db().upsert_websocket_connection(
|
|
||||||
&WebsocketConnection {
|
|
||||||
error: Some(e.to_string()),
|
|
||||||
state: WebsocketConnectionState::Closed,
|
|
||||||
..connection
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
app_handle.db().upsert_websocket_event(
|
|
||||||
&WebsocketEvent {
|
|
||||||
connection_id: connection.id.clone(),
|
|
||||||
request_id: request.id.clone(),
|
|
||||||
workspace_id: connection.workspace_id.clone(),
|
|
||||||
is_server: false,
|
|
||||||
message_type: WebsocketEventType::Open,
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let response_headers = response
|
|
||||||
.headers()
|
|
||||||
.into_iter()
|
|
||||||
.map(|(name, value)| HttpResponseHeader {
|
|
||||||
name: name.to_string(),
|
|
||||||
value: value.to_str().unwrap().to_string(),
|
|
||||||
})
|
|
||||||
.collect::<Vec<HttpResponseHeader>>();
|
|
||||||
|
|
||||||
let connection = app_handle.db().upsert_websocket_connection(
|
|
||||||
&WebsocketConnection {
|
|
||||||
state: WebsocketConnectionState::Connected,
|
|
||||||
headers: response_headers,
|
|
||||||
status: response.status().as_u16() as i32,
|
|
||||||
url: request.url.clone(),
|
|
||||||
..connection
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(window.label()),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
{
|
|
||||||
let connection_id = connection.id.clone();
|
|
||||||
let request_id = request.id.to_string();
|
|
||||||
let workspace_id = request.workspace_id.clone();
|
|
||||||
let connection = connection.clone();
|
|
||||||
let window_label = window.label().to_string();
|
|
||||||
let mut has_written_close = false;
|
|
||||||
tokio::spawn(async move {
|
|
||||||
while let Some(message) = receive_rx.recv().await {
|
|
||||||
if let Message::Close(_) = message {
|
|
||||||
has_written_close = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
app_handle
|
|
||||||
.db()
|
|
||||||
.upsert_websocket_event(
|
|
||||||
&WebsocketEvent {
|
|
||||||
connection_id: connection_id.clone(),
|
|
||||||
request_id: request_id.clone(),
|
|
||||||
workspace_id: workspace_id.clone(),
|
|
||||||
is_server: true,
|
|
||||||
message_type: match message {
|
|
||||||
Message::Text(_) => WebsocketEventType::Text,
|
|
||||||
Message::Binary(_) => WebsocketEventType::Binary,
|
|
||||||
Message::Ping(_) => WebsocketEventType::Ping,
|
|
||||||
Message::Pong(_) => WebsocketEventType::Pong,
|
|
||||||
Message::Close(_) => WebsocketEventType::Close,
|
|
||||||
// Raw frame will never happen during a read
|
|
||||||
Message::Frame(_) => WebsocketEventType::Frame,
|
|
||||||
},
|
|
||||||
message: message.into_data().into(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(&window_label),
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
info!("Websocket connection closed");
|
|
||||||
if !has_written_close {
|
|
||||||
app_handle
|
|
||||||
.db()
|
|
||||||
.upsert_websocket_event(
|
|
||||||
&WebsocketEvent {
|
|
||||||
connection_id: connection_id.clone(),
|
|
||||||
request_id: request_id.clone(),
|
|
||||||
workspace_id: workspace_id.clone(),
|
|
||||||
is_server: true,
|
|
||||||
message_type: WebsocketEventType::Close,
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(&window_label),
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
app_handle
|
|
||||||
.db()
|
|
||||||
.upsert_websocket_connection(
|
|
||||||
&WebsocketConnection {
|
|
||||||
workspace_id: request.workspace_id.clone(),
|
|
||||||
request_id: request_id.to_string(),
|
|
||||||
state: WebsocketConnectionState::Closed,
|
|
||||||
..connection
|
|
||||||
},
|
|
||||||
&UpdateSource::from_window_label(&window_label),
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(connection)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Resolve inherited authentication and headers for a websocket request
|
|
||||||
fn resolve_websocket_request<R: Runtime>(
|
|
||||||
window: &WebviewWindow<R>,
|
|
||||||
request: &WebsocketRequest,
|
|
||||||
) -> Result<(WebsocketRequest, String)> {
|
|
||||||
let mut new_request = request.clone();
|
|
||||||
|
|
||||||
let (authentication_type, authentication, authentication_context_id) =
|
|
||||||
window.db().resolve_auth_for_websocket_request(request)?;
|
|
||||||
new_request.authentication_type = authentication_type;
|
|
||||||
new_request.authentication = authentication;
|
|
||||||
|
|
||||||
let headers = window.db().resolve_headers_for_websocket_request(request)?;
|
|
||||||
new_request.headers = headers;
|
|
||||||
|
|
||||||
Ok((new_request, authentication_context_id))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Convert WS URL to HTTP URL for cookie filtering
|
|
||||||
/// WebSocket upgrade requests are HTTP requests initially, so HttpOnly cookies should apply
|
|
||||||
fn convert_ws_url_to_http(ws_url: &Url) -> Url {
|
|
||||||
let mut http_url = ws_url.clone();
|
|
||||||
|
|
||||||
match ws_url.scheme() {
|
|
||||||
"ws" => {
|
|
||||||
http_url.set_scheme("http").expect("Failed to set http scheme");
|
|
||||||
}
|
|
||||||
"wss" => {
|
|
||||||
http_url.set_scheme("https").expect("Failed to set https scheme");
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
// Already HTTP/HTTPS, no conversion needed
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
http_url
|
|
||||||
}
|
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 10 KiB |
@@ -1,51 +0,0 @@
|
|||||||
{
|
|
||||||
"productName": "Yaak",
|
|
||||||
"version": "0.0.0",
|
|
||||||
"identifier": "app.yaak.desktop",
|
|
||||||
"build": {
|
|
||||||
"beforeBuildCommand": "npm run tauri-before-build",
|
|
||||||
"beforeDevCommand": "npm run tauri-before-dev",
|
|
||||||
"devUrl": "http://localhost:1420",
|
|
||||||
"frontendDist": "../../dist"
|
|
||||||
},
|
|
||||||
"app": {
|
|
||||||
"withGlobalTauri": false,
|
|
||||||
"security": {
|
|
||||||
"assetProtocol": {
|
|
||||||
"enable": true,
|
|
||||||
"scope": {
|
|
||||||
"allow": [
|
|
||||||
"$APPDATA/responses/*",
|
|
||||||
"$RESOURCE/static/*"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"plugins": {
|
|
||||||
"deep-link": {
|
|
||||||
"desktop": {
|
|
||||||
"schemes": [
|
|
||||||
"yaak"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"bundle": {
|
|
||||||
"icon": [
|
|
||||||
"icons/release/32x32.png",
|
|
||||||
"icons/release/128x128.png",
|
|
||||||
"icons/release/128x128@2x.png",
|
|
||||||
"icons/release/icon.icns",
|
|
||||||
"icons/release/icon.ico"
|
|
||||||
],
|
|
||||||
"resources": [
|
|
||||||
"static",
|
|
||||||
"vendored/protoc/include",
|
|
||||||
"vendored/plugins",
|
|
||||||
"vendored/plugin-runtime",
|
|
||||||
"vendored/node/yaaknode*",
|
|
||||||
"vendored/protoc/yaakprotoc*"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
{
|
|
||||||
"build": {
|
|
||||||
"features": ["updater", "license"]
|
|
||||||
},
|
|
||||||
"app": {
|
|
||||||
"security": {
|
|
||||||
"capabilities": [
|
|
||||||
"default",
|
|
||||||
{
|
|
||||||
"identifier": "release",
|
|
||||||
"windows": ["*"],
|
|
||||||
"permissions": ["yaak-license:default"]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"plugins": {
|
|
||||||
"updater": {
|
|
||||||
"endpoints": [
|
|
||||||
"https://update.yaak.app/check/{{target}}/{{arch}}/{{current_version}}"
|
|
||||||
],
|
|
||||||
"pubkey": "dW50cnVzdGVkIGNvbW1lbnQ6IG1pbmlzaWduIHB1YmxpYyBrZXk6IEVGRkFGMjQxRUNEOTQ3MzAKUldRd1I5bnNRZkw2NzRtMnRlWTN3R24xYUR3aGRsUjJzWGwvdHdEcGljb3ZJMUNlMjFsaHlqVU4K"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"bundle": {
|
|
||||||
"publisher": "Yaak",
|
|
||||||
"license": "MIT",
|
|
||||||
"copyright": "Yaak",
|
|
||||||
"homepage": "https://yaak.app",
|
|
||||||
"active": true,
|
|
||||||
"category": "DeveloperTool",
|
|
||||||
"createUpdaterArtifacts": true,
|
|
||||||
"longDescription": "A cross-platform desktop app for interacting with REST, GraphQL, and gRPC",
|
|
||||||
"shortDescription": "Play with APIs, intuitively",
|
|
||||||
"targets": ["app", "appimage", "deb", "dmg", "nsis", "rpm"],
|
|
||||||
"macOS": {
|
|
||||||
"minimumSystemVersion": "13.0",
|
|
||||||
"exceptionDomain": "",
|
|
||||||
"entitlements": "macos/entitlements.plist",
|
|
||||||
"frameworks": []
|
|
||||||
},
|
|
||||||
"windows": {
|
|
||||||
"signCommand": "trusted-signing-cli -e https://eus.codesigning.azure.net/ -a Yaak -c yaakapp %1"
|
|
||||||
},
|
|
||||||
"linux": {
|
|
||||||
"deb": {
|
|
||||||
"desktopTemplate": "./template.desktop",
|
|
||||||
"files": {
|
|
||||||
"/usr/share/metainfo/app.yaak.Yaak.metainfo.xml": "../../flatpak/app.yaak.Yaak.metainfo.xml"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"rpm": {
|
|
||||||
"desktopTemplate": "./template.desktop",
|
|
||||||
"files": {
|
|
||||||
"/usr/share/metainfo/app.yaak.Yaak.metainfo.xml": "../../flatpak/app.yaak.Yaak.metainfo.xml"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
[Desktop Entry]
|
|
||||||
Categories={{categories}}
|
|
||||||
Comment={{comment}}
|
|
||||||
Exec={{exec}}
|
|
||||||
Icon={{icon}}
|
|
||||||
Name={{name}}
|
|
||||||
StartupWMClass={{exec}}
|
|
||||||
Terminal=false
|
|
||||||
Type=Application
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "yaak-fonts"
|
|
||||||
links = "yaak-fonts"
|
|
||||||
version = "0.1.0"
|
|
||||||
edition = "2021"
|
|
||||||
publish = false
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
font-loader = "0.11.0"
|
|
||||||
tauri = { workspace = true }
|
|
||||||
ts-rs = { workspace = true }
|
|
||||||
serde = "1.0"
|
|
||||||
thiserror = { workspace = true }
|
|
||||||
|
|
||||||
[build-dependencies]
|
|
||||||
tauri-plugin = { workspace = true, features = ["build"] }
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.
|
|
||||||
|
|
||||||
export type Fonts = { editorFonts: Array<string>, uiFonts: Array<string>, };
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
const COMMANDS: &[&str] = &["list"];
|
|
||||||
|
|
||||||
fn main() {
|
|
||||||
tauri_plugin::Builder::new(COMMANDS).build();
|
|
||||||
}
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { invoke } from '@tauri-apps/api/core';
|
|
||||||
import { Fonts } from './bindings/gen_fonts';
|
|
||||||
|
|
||||||
export async function listFonts() {
|
|
||||||
return invoke<Fonts>('plugin:yaak-fonts|list', {});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useFonts() {
|
|
||||||
return useQuery({
|
|
||||||
queryKey: ['list_fonts'],
|
|
||||||
queryFn: () => listFonts(),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "@yaakapp-internal/fonts",
|
|
||||||
"private": true,
|
|
||||||
"version": "1.0.0",
|
|
||||||
"main": "index.ts"
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
[default]
|
|
||||||
description = "Default permissions for the plugin"
|
|
||||||
permissions = ["allow-list"]
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
use crate::Result;
|
|
||||||
use font_loader::system_fonts;
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
use std::collections::HashSet;
|
|
||||||
use tauri::command;
|
|
||||||
use ts_rs::TS;
|
|
||||||
|
|
||||||
#[derive(Default, Debug, Clone, Serialize, Deserialize, TS, PartialEq)]
|
|
||||||
#[serde(rename_all = "camelCase")]
|
|
||||||
#[ts(export, export_to = "gen_fonts.ts")]
|
|
||||||
pub struct Fonts {
|
|
||||||
pub editor_fonts: Vec<String>,
|
|
||||||
pub ui_fonts: Vec<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[command]
|
|
||||||
pub(crate) async fn list() -> Result<Fonts> {
|
|
||||||
let mut ui_fonts = HashSet::new();
|
|
||||||
let mut editor_fonts = HashSet::new();
|
|
||||||
|
|
||||||
let mut property = system_fonts::FontPropertyBuilder::new().monospace().build();
|
|
||||||
for font in &system_fonts::query_specific(&mut property) {
|
|
||||||
editor_fonts.insert(font.to_string());
|
|
||||||
}
|
|
||||||
for font in &system_fonts::query_all() {
|
|
||||||
if !editor_fonts.contains(font) {
|
|
||||||
ui_fonts.insert(font.to_string());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut ui_fonts: Vec<String> = ui_fonts.into_iter().collect();
|
|
||||||
let mut editor_fonts: Vec<String> = editor_fonts.into_iter().collect();
|
|
||||||
|
|
||||||
ui_fonts.sort();
|
|
||||||
editor_fonts.sort();
|
|
||||||
|
|
||||||
Ok(Fonts { ui_fonts, editor_fonts })
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
use serde::{ser::Serializer, Serialize};
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error)]
|
|
||||||
pub enum Error {}
|
|
||||||
|
|
||||||
impl Serialize for Error {
|
|
||||||
fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
|
|
||||||
where
|
|
||||||
S: Serializer,
|
|
||||||
{
|
|
||||||
serializer.serialize_str(self.to_string().as_ref())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub type Result<T> = std::result::Result<T, Error>;
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user