24 Commits

Author SHA1 Message Date
67ff31104f chore: Bump version to 0.1.4
- Fix: Skip empty/whitespace text content blocks
- Fix: Validate function call arguments and skip malformed calls
- Fix: Skip outputs for skipped function calls to maintain consistency
- Resolves Anthropic API errors:
  - "messages: text content blocks must contain non-whitespace text"
  - "Extra data: line 1 column 26 (char 25)" (invalid JSON)
  - "unexpected `tool_use_id` found in `tool_result` blocks"

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-14 19:52:13 +01:00
866ca2a372 chore: Bump version to 0.1.3
- Fix: Skip empty/whitespace-only text content blocks in chat completions
- This resolves Anthropic API errors:
  - "messages: text content blocks must contain non-whitespace text"
  - "messages: text content blocks must be non-empty"
- Updated version strings in all test files and snapshots

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-14 14:30:46 +01:00
Sebastian Krüger
4b2e4a1d48 fix: Add scope and update npm for OIDC publishing
- Added scope: "@valknarthing" to setup-node action (required for scoped packages)
- Added npm update step to ensure npm CLI v11.5.1+ (required for OIDC support)

Matches the original OpenAI Codex workflow configuration.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 13:24:05 +01:00
Sebastian Krüger
207a0e2333 fix: Remove NPM_TOKEN for OIDC auth and disable alpha branch update
- Removed NODE_AUTH_TOKEN env var from publish-npm job
  OIDC/Trusted Publishers authentication doesn't need NPM_TOKEN secret
- Commented out update-branch job since latest-alpha-cli branch doesn't exist

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 12:52:52 +01:00
Sebastian Krüger
df6e9f8e0e chore: Bump version to 0.1.2
Updated all version references from 0.1.1 to 0.1.2:
- Workspace version in llmx-rs/Cargo.toml
- Package version in llmx-cli/package.json
- Updated Cargo.lock with all workspace crate versions
- Updated test hardcoded version strings in:
  - mcp-server/tests/common/mcp_process.rs
  - app-server/tests/suite/user_agent.rs
  - app-server/tests/common/mcp_process.rs
- Updated TUI snapshot tests with new version number

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 12:21:18 +01:00
Sebastian Krüger
c9f903a83e fix: Correct environment variables and documentation links in README
- Fixed environment variable names from LITELLM_* to LLMX_*
  - LITELLM_BASE_URL → LLMX_BASE_URL
  - LITELLM_API_KEY → LLMX_API_KEY
- Updated all documentation links to use absolute GitHub URLs instead of relative paths
  - Fixes 404 errors when viewing README on npm registry
  - All ./docs/ and ./LITELLM-SETUP.md links now point to github.com/valknarthing/llmx

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 12:18:31 +01:00
Sebastian Krüger
00eed932c0 feat: Use npm Trusted Publishers (OIDC) for automated publishing
- Add id-token: write permission for OIDC authentication
- Add --provenance flag to npm publish for supply chain security
- Use NODE_AUTH_TOKEN environment variable (set by setup-node)
- Remove manual .npmrc token writing (handled by setup-node with OIDC)

This enables automated npm publishing without storing tokens as secrets.
Requires Trusted Publisher to be configured at:
https://www.npmjs.com/package/@valknarthing/llmx/access

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 12:10:09 +01:00
Sebastian Krüger
b47a4dc354 fix: Add --access public flag to npm publish for scoped packages
Scoped packages (@valknarthing/llmx) require the --access public flag
for all npm publish commands, not just the first one.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 11:36:16 +01:00
Sebastian Krüger
d6f414b0ea fix: Update test version references from 0.1.0 to 0.1.1
Update hardcoded version strings in test files to match the new 0.1.1 version:
- llmx-app-server user_agent test
- llmx-app-server test helper initialize
- llmx-mcp-server test helper initialize
- llmx-tui status snapshot tests (auto-updated via cargo insta)

All previously failing tests now pass.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 11:34:23 +01:00
Sebastian Krüger
5f848fe8be chore: Bump version to 0.1.1
🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 10:58:06 +01:00
Sebastian Krüger
ca37c07257 chore: Remove OpenAI branding and references
Update project branding to reflect community fork status:

- Changed welcome message from "OpenAI's command-line coding agent"
  to "your command-line coding agent"
- Updated system prompt from "led by OpenAI" to "community project"
- Removed broken developers.openai.com/llmx documentation URLs
- Updated Windows setup instructions with correct npm package name
- Removed OpenAI CLA (not applicable to community fork)
- Removed OpenAI open source fund documentation
- Updated config docs to remove managed configuration references
- Changed npm package reference from @openai/llmx to @valknarthing/llmx

These changes complete the rebranding from OpenAI's project to an
independent community fork while maintaining proper attribution to
the original project in README.md.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 10:52:57 +01:00
Sebastian Krüger
e89209d021 fix(tests): use public API for login_and_cancel_chatgpt test
- Use read_stream_until_response_message instead of private read_jsonrpc_message
- Simplify test to accept both success and error outcomes for cancel
- Remove unused imports (JSONRPCMessage, CancelLoginChatGptResponse)

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 09:52:57 +01:00
Sebastian Krüger
4ddd4e078f docs: remove images and Homebrew references, fix GitHub URLs
- Remove .github/*.png images (not needed)
- Remove Homebrew installation instructions (no cask available)
- Fix original project URL to point to openai/codex
- Fix GitHub URLs from valknar to valknarthing

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 09:50:21 +01:00
Sebastian Krüger
5a9de8e195 fix(release): configure release workflow for npm publishing
- Delete existing release before creating new one to avoid conflicts
- Remove llmx-responses-api-proxy from dotslash config (not needed)
- Add 'Create GitHub Release' step to upload platform binaries
- Configure npm authentication by writing token to setup-node's .npmrc
- Use internal package identifier 'llmx' in workflow scripts

This fixes the release workflow to properly:
1. Clean up previous failed releases
2. Create GitHub release with all platform binaries
3. Add DotSlash manifest file (llmx) without conflicts
4. Publish to npm as @valknarthing/llmx with proper authentication

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 09:47:13 +01:00
Sebastian Krüger
89228842fc fix(tests): handle race condition in login_and_cancel_chatgpt test
The test was failing when the login session was cleaned up before the cancel
request could be processed. Now the test accepts both successful cancellation
and 'login id not found' error as valid outcomes.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 08:19:43 +01:00
Sebastian Krüger
edefb6eb9c fix(release): correct package identifier and restore GitHub Release step
- Revert workflow to use internal identifier 'llmx' instead of scoped name
- Revert build_npm_package.py to only accept internal identifiers
- Restore "Create GitHub Release" step that was incorrectly removed
- Package name '@valknarthing/llmx' is read from package.json by the script

This matches OpenAI's approach where they use 'codex' not '@openai/codex' in workflow.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-13 07:50:17 +01:00
Sebastian Krüger
8c04526619 fix(build): accept scoped package name @valknarthing/llmx in build script
The build_npm_package.py script now accepts '@valknarthing/llmx' as a valid
--package argument and normalizes it to 'llmx' internally for processing.

This allows the workflow to use the full scoped package name for clarity
while maintaining backward compatibility with the 'llmx' identifier.
2025-11-13 07:42:12 +01:00
Sebastian Krüger
3821a18ec1 fix(release): remove duplicate GitHub Release creation step
The 'Create GitHub Release' step was conflicting with dotslash-publish-release.
Both were trying to upload assets to the same release, causing the 'asset under
the same name already exists' error.

Removed the softprops/action-gh-release step since facebook/dotslash-publish-release
handles creating the release and uploading all assets including the dotslash manifest.
2025-11-13 07:16:36 +01:00
Sebastian Krüger
40cd73936c fix(release): add clobber flag to dotslash-publish to allow overwriting assets
The dotslash-publish-release step was failing because the 'Create GitHub Release'
step had already uploaded the binary assets. Adding clobber: true allows dotslash
to overwrite/reuse existing assets when creating the dotslash manifest.
2025-11-13 06:44:59 +01:00
Sebastian Krüger
ec0c5a6fb7 feat(release): use NPM_TOKEN for npm publishing instead of OIDC
Changed from npm Trusted Publishing (OIDC) to traditional token authentication.
The NODE_AUTH_TOKEN environment variable now uses the NPM_TOKEN secret.

Removed:
- OIDC id-token permission
- npm CLI update step (no longer needed)

Added:
- NODE_AUTH_TOKEN in both Setup Node.js and Publish to npm steps
2025-11-13 05:19:01 +01:00
Sebastian Krüger
2c0196efd3 chore: rename npm scope from @valknar to @valknarthing
Updated package names across the project:
- @valknar/llmx → @valknarthing/llmx
- @valknar/llmx-sdk → @valknarthing/llmx-sdk
- @valknar/llmx-responses-api-proxy → @valknarthing/llmx-responses-api-proxy

Also updated:
- GitHub repository URLs (valknar → valknarthing)
- GitHub Actions workflow scope configuration
- README installation instructions
2025-11-13 04:59:52 +01:00
Sebastian Krüger
f58398dfbb fix(release): correct npm tarball filename pattern
The stage_npm_packages.py script creates tarballs named {package}-npm-{version}.tgz
(e.g., llmx-npm-0.1.0.tgz), but the workflow was looking for valknar-llmx-npm-{version}.tgz.

Fixed the download pattern and publish tarball list to match the actual output from
the staging script.
2025-11-13 04:37:20 +01:00
Sebastian Krüger
91ce3a3838 fix(release): use correct package name 'llmx' instead of '@valknar/llmx'
The build_npm_package.py script only accepts specific package choices:
'llmx', 'llmx-responses-api-proxy', 'llmx-sdk'. The scoped package name
'@valknar/llmx' is not a valid choice.

The package name 'llmx' is the correct identifier used internally by the
build system, which then produces the scoped npm package.
2025-11-13 04:34:24 +01:00
Sebastian Krüger
f3a1034d5d fix(release): sanitize package name in stage_npm_packages.py for temp directory creation
The script was failing when creating temporary directories for scoped packages like @valknar/llmx
because the forward slash in the package name was being used directly in the temp directory prefix,
causing Python's tempfile.mkdtemp() to fail with 'No such file or directory'.

Fix by sanitizing the package name: replace '/' with '-' and remove '@' before using it in the
temp directory prefix. The actual package output file still uses the original package name.

Fixes: FileNotFoundError in GitHub Actions rust-release workflow
2025-11-13 03:59:25 +01:00
31 changed files with 192 additions and 981 deletions

View File

@@ -27,34 +27,6 @@
"path": "llmx.exe"
}
}
},
"llmx-responses-api-proxy": {
"platforms": {
"macos-aarch64": {
"regex": "^llmx-responses-api-proxy-aarch64-apple-darwin\\.zst$",
"path": "llmx-responses-api-proxy"
},
"macos-x86_64": {
"regex": "^llmx-responses-api-proxy-x86_64-apple-darwin\\.zst$",
"path": "llmx-responses-api-proxy"
},
"linux-x86_64": {
"regex": "^llmx-responses-api-proxy-x86_64-unknown-linux-musl\\.zst$",
"path": "llmx-responses-api-proxy"
},
"linux-aarch64": {
"regex": "^llmx-responses-api-proxy-aarch64-unknown-linux-musl\\.zst$",
"path": "llmx-responses-api-proxy"
},
"windows-x86_64": {
"regex": "^llmx-responses-api-proxy-x86_64-pc-windows-msvc\\.exe\\.zst$",
"path": "llmx-responses-api-proxy.exe"
},
"windows-aarch64": {
"regex": "^llmx-responses-api-proxy-aarch64-pc-windows-msvc\\.exe\\.zst$",
"path": "llmx-responses-api-proxy.exe"
}
}
}
}
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.9 MiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 408 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.1 MiB

View File

@@ -445,7 +445,19 @@ jobs:
run: |
./scripts/stage_npm_packages.py \
--release-version "${{ steps.release_name.outputs.name }}" \
--package @valknar/llmx
--package llmx
# Delete any existing release to avoid conflicts with dotslash manifest file
- name: Delete existing release if present
env:
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
run: |
if gh release view "${{ github.ref_name }}" --repo "${{ github.repository }}" >/dev/null 2>&1; then
echo "Deleting existing release ${{ github.ref_name }}"
gh release delete "${{ github.ref_name }}" --repo "${{ github.repository }}" --yes
else
echo "No existing release found for ${{ github.ref_name }}"
fi
- name: Create GitHub Release
uses: softprops/action-gh-release@v2
@@ -464,9 +476,7 @@ jobs:
tag: ${{ github.ref_name }}
config: .github/dotslash-config.json
# Publish to npm using OIDC authentication.
# July 31, 2025: https://github.blog/changelog/2025-07-31-npm-trusted-publishing-with-oidc-is-generally-available/
# npm docs: https://docs.npmjs.com/trusted-publishers
# Publish to npm using Trusted Publishers (OIDC)
publish-npm:
# Publish to npm for stable releases and alpha pre-releases with numeric suffixes.
if: ${{ needs.release.outputs.should_publish_npm == 'true' }}
@@ -474,8 +484,8 @@ jobs:
needs: release
runs-on: ubuntu-latest
permissions:
id-token: write # Required for OIDC
contents: read
id-token: write # Required for OIDC authentication
steps:
- name: Setup Node.js
@@ -483,9 +493,8 @@ jobs:
with:
node-version: 22
registry-url: "https://registry.npmjs.org"
scope: "@valknar"
scope: "@valknarthing"
# Trusted publishing requires npm CLI version 11.5.1 or later.
- name: Update npm
run: npm install -g npm@latest
@@ -499,10 +508,9 @@ jobs:
mkdir -p dist/npm
gh release download "$tag" \
--repo "${GITHUB_REPOSITORY}" \
--pattern "valknar-llmx-npm-${version}.tgz" \
--pattern "llmx-npm-${version}.tgz" \
--dir dist/npm
# No NODE_AUTH_TOKEN needed because we use OIDC.
- name: Publish to npm
env:
VERSION: ${{ needs.release.outputs.version }}
@@ -515,28 +523,28 @@ jobs:
fi
tarballs=(
"valknar-llmx-npm-${VERSION}.tgz"
"llmx-npm-${VERSION}.tgz"
)
for tarball in "${tarballs[@]}"; do
npm publish "${GITHUB_WORKSPACE}/dist/npm/${tarball}" "${tag_args[@]}"
npm publish "${GITHUB_WORKSPACE}/dist/npm/${tarball}" --provenance --access public "${tag_args[@]}"
done
update-branch:
name: Update latest-alpha-cli branch
permissions:
contents: write
needs: release
runs-on: ubuntu-latest
steps:
- name: Update latest-alpha-cli branch
env:
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
run: |
set -euo pipefail
gh api \
repos/${GITHUB_REPOSITORY}/git/refs/heads/latest-alpha-cli \
-X PATCH \
-f sha="${GITHUB_SHA}" \
-F force=true
# update-branch:
# name: Update latest-alpha-cli branch
# permissions:
# contents: write
# needs: release
# runs-on: ubuntu-latest
#
# steps:
# - name: Update latest-alpha-cli branch
# env:
# GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
# run: |
# set -euo pipefail
# gh api \
# repos/${GITHUB_REPOSITORY}/git/refs/heads/latest-alpha-cli \
# -X PATCH \
# -f sha="${GITHUB_SHA}" \
# -F force=true

View File

@@ -1,13 +1,9 @@
<p align="center"><code>npm i -g @valknar/llmx</code><br />or <code>brew install --cask llmx</code></p>
<p align="center"><code>npm i -g @valknarthing/llmx</code></p>
<p align="center"><strong>LLMX CLI</strong> is a coding agent powered by LiteLLM that runs locally on your computer.
</br>
</br>This project is a community fork with enhanced support for multiple LLM providers via LiteLLM.
</br>Original project: <a href="https://github.com/openai/llmx">github.com/openai/llmx</a></p>
<p align="center">
<img src="./.github/llmx-cli-splash.png" alt="LLMX CLI splash" width="80%" />
</p>
</br>Original project: <a href="https://github.com/openai/codex">github.com/openai/codex</a></p>
---
@@ -15,16 +11,10 @@
### Installing and running LLMX CLI
Install globally with your preferred package manager. If you use npm:
Install globally with npm:
```shell
npm install -g @valknar/llmx
```
Alternatively, if you use Homebrew:
```shell
brew install --cask llmx
npm install -g @valknarthing/llmx
```
Then simply run `llmx` to get started:
@@ -33,10 +23,8 @@ Then simply run `llmx` to get started:
llmx
```
If you're running into upgrade issues with Homebrew, see the [FAQ entry on brew upgrade llmx](./docs/faq.md#brew-upgrade-llmx-isnt-upgrading-me).
<details>
<summary>You can also go to the <a href="https://github.com/valknar/llmx/releases/latest">latest GitHub Release</a> and download the appropriate binary for your platform.</summary>
<summary>You can also go to the <a href="https://github.com/valknarthing/llmx/releases/latest">latest GitHub Release</a> and download the appropriate binary for your platform.</summary>
Each GitHub Release contains many executables, but in practice, you likely want one of these:
@@ -59,57 +47,56 @@ LLMX is powered by [LiteLLM](https://docs.litellm.ai/), which provides access to
```bash
# Set your LiteLLM server URL (default: http://localhost:4000/v1)
export LITELLM_BASE_URL="http://localhost:4000/v1"
export LITELLM_API_KEY="your-api-key"
export LLMX_BASE_URL="http://localhost:4000/v1"
export LLMX_API_KEY="your-api-key"
# Run LLMX
llmx "hello world"
```
**Configuration:** See [LITELLM-SETUP.md](./LITELLM-SETUP.md) for detailed setup instructions.
**Configuration:** See [LITELLM-SETUP.md](https://github.com/valknarthing/llmx/blob/main/LITELLM-SETUP.md) for detailed setup instructions.
You can also use LLMX with ChatGPT or OpenAI API keys. For authentication options, see the [authentication docs](./docs/authentication.md).
You can also use LLMX with ChatGPT or OpenAI API keys. For authentication options, see the [authentication docs](https://github.com/valknarthing/llmx/blob/main/docs/authentication.md).
### Model Context Protocol (MCP)
LLMX can access MCP servers. To configure them, refer to the [config docs](./docs/config.md#mcp_servers).
LLMX can access MCP servers. To configure them, refer to the [config docs](https://github.com/valknarthing/llmx/blob/main/docs/config.md#mcp_servers).
### Configuration
LLMX CLI supports a rich set of configuration options, with preferences stored in `~/.llmx/config.toml`. For full configuration options, see [Configuration](./docs/config.md).
LLMX CLI supports a rich set of configuration options, with preferences stored in `~/.llmx/config.toml`. For full configuration options, see [Configuration](https://github.com/valknarthing/llmx/blob/main/docs/config.md).
---
### Docs & FAQ
- [**Getting started**](./docs/getting-started.md)
- [CLI usage](./docs/getting-started.md#cli-usage)
- [Slash Commands](./docs/slash_commands.md)
- [Running with a prompt as input](./docs/getting-started.md#running-with-a-prompt-as-input)
- [Example prompts](./docs/getting-started.md#example-prompts)
- [Custom prompts](./docs/prompts.md)
- [Memory with AGENTS.md](./docs/getting-started.md#memory-with-agentsmd)
- [**Configuration**](./docs/config.md)
- [Example config](./docs/example-config.md)
- [**Sandbox & approvals**](./docs/sandbox.md)
- [**Authentication**](./docs/authentication.md)
- [Auth methods](./docs/authentication.md#forcing-a-specific-auth-method-advanced)
- [Login on a "Headless" machine](./docs/authentication.md#connecting-on-a-headless-machine)
- [**Getting started**](https://github.com/valknarthing/llmx/blob/main/docs/getting-started.md)
- [CLI usage](https://github.com/valknarthing/llmx/blob/main/docs/getting-started.md#cli-usage)
- [Slash Commands](https://github.com/valknarthing/llmx/blob/main/docs/slash_commands.md)
- [Running with a prompt as input](https://github.com/valknarthing/llmx/blob/main/docs/getting-started.md#running-with-a-prompt-as-input)
- [Example prompts](https://github.com/valknarthing/llmx/blob/main/docs/getting-started.md#example-prompts)
- [Custom prompts](https://github.com/valknarthing/llmx/blob/main/docs/prompts.md)
- [Memory with AGENTS.md](https://github.com/valknarthing/llmx/blob/main/docs/getting-started.md#memory-with-agentsmd)
- [**Configuration**](https://github.com/valknarthing/llmx/blob/main/docs/config.md)
- [Example config](https://github.com/valknarthing/llmx/blob/main/docs/example-config.md)
- [**Sandbox & approvals**](https://github.com/valknarthing/llmx/blob/main/docs/sandbox.md)
- [**Authentication**](https://github.com/valknarthing/llmx/blob/main/docs/authentication.md)
- [Auth methods](https://github.com/valknarthing/llmx/blob/main/docs/authentication.md#forcing-a-specific-auth-method-advanced)
- [Login on a "Headless" machine](https://github.com/valknarthing/llmx/blob/main/docs/authentication.md#connecting-on-a-headless-machine)
- **Automating LLMX**
- [GitHub Action](https://github.com/valknar/llmx-action)
- [TypeScript SDK](./sdk/typescript/README.md)
- [Non-interactive mode (`llmx exec`)](./docs/exec.md)
- [**Advanced**](./docs/advanced.md)
- [Tracing / verbose logging](./docs/advanced.md#tracing--verbose-logging)
- [Model Context Protocol (MCP)](./docs/advanced.md#model-context-protocol-mcp)
- [**Zero data retention (ZDR)**](./docs/zdr.md)
- [**Contributing**](./docs/contributing.md)
- [**Install & build**](./docs/install.md)
- [System Requirements](./docs/install.md#system-requirements)
- [DotSlash](./docs/install.md#dotslash)
- [Build from source](./docs/install.md#build-from-source)
- [**FAQ**](./docs/faq.md)
- [**Open source fund**](./docs/open-source-fund.md)
- [GitHub Action](https://github.com/valknarthing/llmx-action)
- [TypeScript SDK](https://github.com/valknarthing/llmx/blob/main/sdk/typescript/README.md)
- [Non-interactive mode (`llmx exec`)](https://github.com/valknarthing/llmx/blob/main/docs/exec.md)
- [**Advanced**](https://github.com/valknarthing/llmx/blob/main/docs/advanced.md)
- [Tracing / verbose logging](https://github.com/valknarthing/llmx/blob/main/docs/advanced.md#tracing--verbose-logging)
- [Model Context Protocol (MCP)](https://github.com/valknarthing/llmx/blob/main/docs/advanced.md#model-context-protocol-mcp)
- [**Zero data retention (ZDR)**](https://github.com/valknarthing/llmx/blob/main/docs/zdr.md)
- [**Contributing**](https://github.com/valknarthing/llmx/blob/main/docs/contributing.md)
- [**Install & build**](https://github.com/valknarthing/llmx/blob/main/docs/install.md)
- [System Requirements](https://github.com/valknarthing/llmx/blob/main/docs/install.md#system-requirements)
- [DotSlash](https://github.com/valknarthing/llmx/blob/main/docs/install.md#dotslash)
- [Build from source](https://github.com/valknarthing/llmx/blob/main/docs/install.md#build-from-source)
- [**FAQ**](https://github.com/valknarthing/llmx/blob/main/docs/faq.md)
---

View File

@@ -1,49 +0,0 @@
# Individual Contributor License Agreement (v1.0, OpenAI)
_Based on the Apache Software Foundation Individual CLA v 2.2._
By commenting **“I have read the CLA Document and I hereby sign the CLA”**
on a Pull Request, **you (“Contributor”) agree to the following terms** for any
past and future “Contributions” submitted to the **OpenAI LLMX CLI project
(the “Project”)**.
---
## 1. Definitions
- **“Contribution”** any original work of authorship submitted to the Project
(code, documentation, designs, etc.).
- **“You” / “Your”** the individual (or legal entity) posting the acceptance
comment.
## 2. Copyright License
You grant **OpenAI, Inc.** and all recipients of software distributed by the
Project a perpetual, worldwide, nonexclusive, royaltyfree, irrevocable
license to reproduce, prepare derivative works of, publicly display, publicly
perform, sublicense, and distribute Your Contributions and derivative works.
## 3. Patent License
You grant **OpenAI, Inc.** and all recipients of the Project a perpetual,
worldwide, nonexclusive, royaltyfree, irrevocable (except as below) patent
license to make, have made, use, sell, offer to sell, import, and otherwise
transfer Your Contributions alone or in combination with the Project.
If any entity brings patent litigation alleging that the Project or a
Contribution infringes a patent, the patent licenses granted by You to that
entity under this CLA terminate.
## 4. Representations
1. You are legally entitled to grant the licenses above.
2. Each Contribution is either Your original creation or You have authority to
submit it under this CLA.
3. Your Contributions are provided **“AS IS”** without warranties of any kind.
4. You will notify the Project if any statement above becomes inaccurate.
## 5. Miscellany
This Agreement is governed by the laws of the **State of California**, USA,
excluding its conflictoflaws rules. If any provision is held unenforceable,
the remaining provisions remain in force.

View File

@@ -873,7 +873,7 @@ notifications = [ "agent-turn-complete", "approval-requested" ]
### Forcing a login method
To force users on a given machine to use a specific login method or workspace, use a combination of [managed configurations](https://developers.openai.com/llmx/security#managed-configuration) as well as either or both of the following fields:
To force users on a given machine to use a specific login method or workspace, use either or both of the following fields in your configuration:
```toml
# Force the user to log in with ChatGPT or via an api key.

View File

@@ -1,8 +0,0 @@
## LLMX open source fund
We're excited to launch a **$1 million initiative** supporting open source projects that use LLMX CLI and other OpenAI models.
- Grants are awarded up to **$25,000** API credits.
- Applications are reviewed **on a rolling basis**.
**Interested? [Apply here](https://openai.com/form/llmx-open-source-fund/).**

View File

@@ -1,736 +0,0 @@
<h1 align="center">OpenAI LLMX CLI</h1>
<p align="center">Lightweight coding agent that runs in your terminal</p>
<p align="center"><code>npm i -g @llmx/llmx</code></p>
> [!IMPORTANT]
> This is the documentation for the _legacy_ TypeScript implementation of the LLMX CLI. It has been superseded by the _Rust_ implementation. See the [README in the root of the LLMX repository](https://github.com/valknar/llmx/blob/main/README.md) for details.
![LLMX demo GIF using: llmx "explain this codebase to me"](../.github/demo.gif)
---
<details>
<summary><strong>Table of contents</strong></summary>
<!-- Begin ToC -->
- [Experimental technology disclaimer](#experimental-technology-disclaimer)
- [Quickstart](#quickstart)
- [Why LLMX?](#why-llmx)
- [Security model & permissions](#security-model--permissions)
- [Platform sandboxing details](#platform-sandboxing-details)
- [System requirements](#system-requirements)
- [CLI reference](#cli-reference)
- [Memory & project docs](#memory--project-docs)
- [Non-interactive / CI mode](#non-interactive--ci-mode)
- [Tracing / verbose logging](#tracing--verbose-logging)
- [Recipes](#recipes)
- [Installation](#installation)
- [Configuration guide](#configuration-guide)
- [Basic configuration parameters](#basic-configuration-parameters)
- [Custom AI provider configuration](#custom-ai-provider-configuration)
- [History configuration](#history-configuration)
- [Configuration examples](#configuration-examples)
- [Full configuration example](#full-configuration-example)
- [Custom instructions](#custom-instructions)
- [Environment variables setup](#environment-variables-setup)
- [FAQ](#faq)
- [Zero data retention (ZDR) usage](#zero-data-retention-zdr-usage)
- [LLMX open source fund](#llmx-open-source-fund)
- [Contributing](#contributing)
- [Development workflow](#development-workflow)
- [Git hooks with Husky](#git-hooks-with-husky)
- [Debugging](#debugging)
- [Writing high-impact code changes](#writing-high-impact-code-changes)
- [Opening a pull request](#opening-a-pull-request)
- [Review process](#review-process)
- [Community values](#community-values)
- [Getting help](#getting-help)
- [Contributor license agreement (CLA)](#contributor-license-agreement-cla)
- [Quick fixes](#quick-fixes)
- [Releasing `llmx`](#releasing-llmx)
- [Alternative build options](#alternative-build-options)
- [Nix flake development](#nix-flake-development)
- [Security & responsible AI](#security--responsible-ai)
- [License](#license)
<!-- End ToC -->
</details>
---
## Experimental technology disclaimer
LLMX CLI is an experimental project under active development. It is not yet stable, may contain bugs, incomplete features, or undergo breaking changes. We're building it in the open with the community and welcome:
- Bug reports
- Feature requests
- Pull requests
- Good vibes
Help us improve by filing issues or submitting PRs (see the section below for how to contribute)!
## Quickstart
Install globally:
```shell
npm install -g @llmx/llmx
```
Next, set your OpenAI API key as an environment variable:
```shell
export OPENAI_API_KEY="your-api-key-here"
```
> **Note:** This command sets the key only for your current terminal session. You can add the `export` line to your shell's configuration file (e.g., `~/.zshrc`) but we recommend setting for the session. **Tip:** You can also place your API key into a `.env` file at the root of your project:
>
> ```env
> OPENAI_API_KEY=your-api-key-here
> ```
>
> The CLI will automatically load variables from `.env` (via `dotenv/config`).
<details>
<summary><strong>Use <code>--provider</code> to use other models</strong></summary>
> LLMX also allows you to use other providers that support the OpenAI Chat Completions API. You can set the provider in the config file or use the `--provider` flag. The possible options for `--provider` are:
>
> - openai (default)
> - openrouter
> - azure
> - gemini
> - ollama
> - mistral
> - deepseek
> - xai
> - groq
> - arceeai
> - any other provider that is compatible with the OpenAI API
>
> If you use a provider other than OpenAI, you will need to set the API key for the provider in the config file or in the environment variable as:
>
> ```shell
> export <provider>_API_KEY="your-api-key-here"
> ```
>
> If you use a provider not listed above, you must also set the base URL for the provider:
>
> ```shell
> export <provider>_BASE_URL="https://your-provider-api-base-url"
> ```
</details>
<br />
Run interactively:
```shell
llmx
```
Or, run with a prompt as input (and optionally in `Full Auto` mode):
```shell
llmx "explain this codebase to me"
```
```shell
llmx --approval-mode full-auto "create the fanciest todo-list app"
```
That's it - LLMX will scaffold a file, run it inside a sandbox, install any
missing dependencies, and show you the live result. Approve the changes and
they'll be committed to your working directory.
---
## Why LLMX?
LLMX CLI is built for developers who already **live in the terminal** and want
ChatGPT-level reasoning **plus** the power to actually run code, manipulate
files, and iterate - all under version control. In short, it's _chat-driven
development_ that understands and executes your repo.
- **Zero setup** - bring your OpenAI API key and it just works!
- **Full auto-approval, while safe + secure** by running network-disabled and directory-sandboxed
- **Multimodal** - pass in screenshots or diagrams to implement features ✨
And it's **fully open-source** so you can see and contribute to how it develops!
---
## Security model & permissions
LLMX lets you decide _how much autonomy_ the agent receives and auto-approval policy via the
`--approval-mode` flag (or the interactive onboarding prompt):
| Mode | What the agent may do without asking | Still requires approval |
| ------------------------- | --------------------------------------------------------------------------------------------------- | ----------------------------------------------------------------------------------------------- |
| **Suggest** <br>(default) | <li>Read any file in the repo | <li>**All** file writes/patches<li> **Any** arbitrary shell commands (aside from reading files) |
| **Auto Edit** | <li>Read **and** apply-patch writes to files | <li>**All** shell commands |
| **Full Auto** | <li>Read/write files <li> Execute shell commands (network disabled, writes limited to your workdir) | - |
In **Full Auto** every command is run **network-disabled** and confined to the
current working directory (plus temporary files) for defense-in-depth. LLMX
will also show a warning/confirmation if you start in **auto-edit** or
**full-auto** while the directory is _not_ tracked by Git, so you always have a
safety net.
Coming soon: you'll be able to whitelist specific commands to auto-execute with
the network enabled, once we're confident in additional safeguards.
### Platform sandboxing details
The hardening mechanism LLMX uses depends on your OS:
- **macOS 12+** - commands are wrapped with **Apple Seatbelt** (`sandbox-exec`).
- Everything is placed in a read-only jail except for a small set of
writable roots (`$PWD`, `$TMPDIR`, `~/.llmx`, etc.).
- Outbound network is _fully blocked_ by default - even if a child process
tries to `curl` somewhere it will fail.
- **Linux** - there is no sandboxing by default.
We recommend using Docker for sandboxing, where LLMX launches itself inside a **minimal
container image** and mounts your repo _read/write_ at the same path. A
custom `iptables`/`ipset` firewall script denies all egress except the
OpenAI API. This gives you deterministic, reproducible runs without needing
root on the host. You can use the [`run_in_container.sh`](../llmx-cli/scripts/run_in_container.sh) script to set up the sandbox.
---
## System requirements
| Requirement | Details |
| --------------------------- | --------------------------------------------------------------- |
| Operating systems | macOS 12+, Ubuntu 20.04+/Debian 10+, or Windows 11 **via WSL2** |
| Node.js | **16 or newer** (Node 20 LTS recommended) |
| Git (optional, recommended) | 2.23+ for built-in PR helpers |
| RAM | 4-GB minimum (8-GB recommended) |
> Never run `sudo npm install -g`; fix npm permissions instead.
---
## CLI reference
| Command | Purpose | Example |
| ------------------------------------ | ----------------------------------- | ------------------------------------ |
| `llmx` | Interactive REPL | `llmx` |
| `llmx "..."` | Initial prompt for interactive REPL | `llmx "fix lint errors"` |
| `llmx -q "..."` | Non-interactive "quiet mode" | `llmx -q --json "explain utils.ts"` |
| `llmx completion <bash\|zsh\|fish>` | Print shell completion script | `llmx completion bash` |
Key flags: `--model/-m`, `--approval-mode/-a`, `--quiet/-q`, and `--notify`.
---
## Memory & project docs
You can give LLMX extra instructions and guidance using `AGENTS.md` files. LLMX looks for `AGENTS.md` files in the following places, and merges them top-down:
1. `~/.llmx/AGENTS.md` - personal global guidance
2. `AGENTS.md` at repo root - shared project notes
3. `AGENTS.md` in the current working directory - sub-folder/feature specifics
Disable loading of these files with `--no-project-doc` or the environment variable `LLMX_DISABLE_PROJECT_DOC=1`.
---
## Non-interactive / CI mode
Run LLMX head-less in pipelines. Example GitHub Action step:
```yaml
- name: Update changelog via LLMX
run: |
npm install -g @llmx/llmx
export OPENAI_API_KEY="${{ secrets.OPENAI_KEY }}"
llmx -a auto-edit --quiet "update CHANGELOG for next release"
```
Set `LLMX_QUIET_MODE=1` to silence interactive UI noise.
## Tracing / verbose logging
Setting the environment variable `DEBUG=true` prints full API request and response details:
```shell
DEBUG=true llmx
```
---
## Recipes
Below are a few bite-size examples you can copy-paste. Replace the text in quotes with your own task. See the [prompting guide](https://github.com/valknar/llmx/blob/main/llmx-cli/examples/prompting_guide.md) for more tips and usage patterns.
| ✨ | What you type | What happens |
| --- | ------------------------------------------------------------------------------- | -------------------------------------------------------------------------- |
| 1 | `llmx "Refactor the Dashboard component to React Hooks"` | LLMX rewrites the class component, runs `npm test`, and shows the diff. |
| 2 | `llmx "Generate SQL migrations for adding a users table"` | Infers your ORM, creates migration files, and runs them in a sandboxed DB. |
| 3 | `llmx "Write unit tests for utils/date.ts"` | Generates tests, executes them, and iterates until they pass. |
| 4 | `llmx "Bulk-rename *.jpeg -> *.jpg with git mv"` | Safely renames files and updates imports/usages. |
| 5 | `llmx "Explain what this regex does: ^(?=.*[A-Z]).{8,}$"` | Outputs a step-by-step human explanation. |
| 6 | `llmx "Carefully review this repo, and propose 3 high impact well-scoped PRs"` | Suggests impactful PRs in the current codebase. |
| 7 | `llmx "Look for vulnerabilities and create a security review report"` | Finds and explains security bugs. |
---
## Installation
<details open>
<summary><strong>From npm (Recommended)</strong></summary>
```bash
npm install -g @llmx/llmx
# or
yarn global add @llmx/llmx
# or
bun install -g @llmx/llmx
# or
pnpm add -g @llmx/llmx
```
</details>
<details>
<summary><strong>Build from source</strong></summary>
```bash
# Clone the repository and navigate to the CLI package
git clone https://github.com/valknar/llmx.git
cd llmx/llmx-cli
# Enable corepack
corepack enable
# Install dependencies and build
pnpm install
pnpm build
# Linux-only: download prebuilt sandboxing binaries (requires gh and zstd).
./scripts/install_native_deps.sh
# Get the usage and the options
node ./dist/cli.js --help
# Run the locally-built CLI directly
node ./dist/cli.js
# Or link the command globally for convenience
pnpm link
```
</details>
---
## Configuration guide
LLMX configuration files can be placed in the `~/.llmx/` directory, supporting both YAML and JSON formats.
### Basic configuration parameters
| Parameter | Type | Default | Description | Available Options |
| ------------------- | ------- | ---------- | -------------------------------- | ---------------------------------------------------------------------------------------------- |
| `model` | string | `o4-mini` | AI model to use | Any model name supporting OpenAI API |
| `approvalMode` | string | `suggest` | AI assistant's permission mode | `suggest` (suggestions only)<br>`auto-edit` (automatic edits)<br>`full-auto` (fully automatic) |
| `fullAutoErrorMode` | string | `ask-user` | Error handling in full-auto mode | `ask-user` (prompt for user input)<br>`ignore-and-continue` (ignore and proceed) |
| `notify` | boolean | `true` | Enable desktop notifications | `true`/`false` |
### Custom AI provider configuration
In the `providers` object, you can configure multiple AI service providers. Each provider requires the following parameters:
| Parameter | Type | Description | Example |
| --------- | ------ | --------------------------------------- | ----------------------------- |
| `name` | string | Display name of the provider | `"OpenAI"` |
| `baseURL` | string | API service URL | `"https://api.openai.com/v1"` |
| `envKey` | string | Environment variable name (for API key) | `"OPENAI_API_KEY"` |
### History configuration
In the `history` object, you can configure conversation history settings:
| Parameter | Type | Description | Example Value |
| ------------------- | ------- | ------------------------------------------------------ | ------------- |
| `maxSize` | number | Maximum number of history entries to save | `1000` |
| `saveHistory` | boolean | Whether to save history | `true` |
| `sensitivePatterns` | array | Patterns of sensitive information to filter in history | `[]` |
### Configuration examples
1. YAML format (save as `~/.llmx/config.yaml`):
```yaml
model: o4-mini
approvalMode: suggest
fullAutoErrorMode: ask-user
notify: true
```
2. JSON format (save as `~/.llmx/config.json`):
```json
{
"model": "o4-mini",
"approvalMode": "suggest",
"fullAutoErrorMode": "ask-user",
"notify": true
}
```
### Full configuration example
Below is a comprehensive example of `config.json` with multiple custom providers:
```json
{
"model": "o4-mini",
"provider": "openai",
"providers": {
"openai": {
"name": "OpenAI",
"baseURL": "https://api.openai.com/v1",
"envKey": "OPENAI_API_KEY"
},
"azure": {
"name": "AzureOpenAI",
"baseURL": "https://YOUR_PROJECT_NAME.openai.azure.com/openai",
"envKey": "AZURE_OPENAI_API_KEY"
},
"openrouter": {
"name": "OpenRouter",
"baseURL": "https://openrouter.ai/api/v1",
"envKey": "OPENROUTER_API_KEY"
},
"gemini": {
"name": "Gemini",
"baseURL": "https://generativelanguage.googleapis.com/v1beta/openai",
"envKey": "GEMINI_API_KEY"
},
"ollama": {
"name": "Ollama",
"baseURL": "http://localhost:11434/v1",
"envKey": "OLLAMA_API_KEY"
},
"mistral": {
"name": "Mistral",
"baseURL": "https://api.mistral.ai/v1",
"envKey": "MISTRAL_API_KEY"
},
"deepseek": {
"name": "DeepSeek",
"baseURL": "https://api.deepseek.com",
"envKey": "DEEPSEEK_API_KEY"
},
"xai": {
"name": "xAI",
"baseURL": "https://api.x.ai/v1",
"envKey": "XAI_API_KEY"
},
"groq": {
"name": "Groq",
"baseURL": "https://api.groq.com/openai/v1",
"envKey": "GROQ_API_KEY"
},
"arceeai": {
"name": "ArceeAI",
"baseURL": "https://conductor.arcee.ai/v1",
"envKey": "ARCEEAI_API_KEY"
}
},
"history": {
"maxSize": 1000,
"saveHistory": true,
"sensitivePatterns": []
}
}
```
### Custom instructions
You can create a `~/.llmx/AGENTS.md` file to define custom guidance for the agent:
```markdown
- Always respond with emojis
- Only use git commands when explicitly requested
```
### Environment variables setup
For each AI provider, you need to set the corresponding API key in your environment variables. For example:
```bash
# OpenAI
export OPENAI_API_KEY="your-api-key-here"
# Azure OpenAI
export AZURE_OPENAI_API_KEY="your-azure-api-key-here"
export AZURE_OPENAI_API_VERSION="2025-04-01-preview" (Optional)
# OpenRouter
export OPENROUTER_API_KEY="your-openrouter-key-here"
# Similarly for other providers
```
---
## FAQ
<details>
<summary>OpenAI released a model called LLMX in 2021 - is this related?</summary>
In 2021, OpenAI released LLMX, an AI system designed to generate code from natural language prompts. That original LLMX model was deprecated as of March 2023 and is separate from the CLI tool.
</details>
<details>
<summary>Which models are supported?</summary>
Any model available with [Responses API](https://platform.openai.com/docs/api-reference/responses). The default is `o4-mini`, but pass `--model gpt-4.1` or set `model: gpt-4.1` in your config file to override.
</details>
<details>
<summary>Why does <code>o3</code> or <code>o4-mini</code> not work for me?</summary>
It's possible that your [API account needs to be verified](https://help.openai.com/en/articles/10910291-api-organization-verification) in order to start streaming responses and seeing chain of thought summaries from the API. If you're still running into issues, please let us know!
</details>
<details>
<summary>How do I stop LLMX from editing my files?</summary>
LLMX runs model-generated commands in a sandbox. If a proposed command or file change doesn't look right, you can simply type **n** to deny the command or give the model feedback.
</details>
<details>
<summary>Does it work on Windows?</summary>
Not directly. It requires [Windows Subsystem for Linux (WSL2)](https://learn.microsoft.com/en-us/windows/wsl/install) - LLMX is regularly tested on macOS and Linux with Node 20+, and also supports Node 16.
</details>
---
## Zero data retention (ZDR) usage
LLMX CLI **does** support OpenAI organizations with [Zero Data Retention (ZDR)](https://platform.openai.com/docs/guides/your-data#zero-data-retention) enabled. If your OpenAI organization has Zero Data Retention enabled and you still encounter errors such as:
```
OpenAI rejected the request. Error details: Status: 400, Code: unsupported_parameter, Type: invalid_request_error, Message: 400 Previous response cannot be used for this organization due to Zero Data Retention.
```
You may need to upgrade to a more recent version with: `npm i -g @llmx/llmx@latest`
---
## LLMX open source fund
We're excited to launch a **$1 million initiative** supporting open source projects that use LLMX CLI and other OpenAI models.
- Grants are awarded up to **$25,000** API credits.
- Applications are reviewed **on a rolling basis**.
**Interested? [Apply here](https://openai.com/form/llmx-open-source-fund/).**
---
## Contributing
This project is under active development and the code will likely change pretty significantly. We'll update this message once that's complete!
More broadly we welcome contributions - whether you are opening your very first pull request or you're a seasoned maintainer. At the same time we care about reliability and long-term maintainability, so the bar for merging code is intentionally **high**. The guidelines below spell out what "high-quality" means in practice and should make the whole process transparent and friendly.
### Development workflow
- Create a _topic branch_ from `main` - e.g. `feat/interactive-prompt`.
- Keep your changes focused. Multiple unrelated fixes should be opened as separate PRs.
- Use `pnpm test:watch` during development for super-fast feedback.
- We use **Vitest** for unit tests, **ESLint** + **Prettier** for style, and **TypeScript** for type-checking.
- Before pushing, run the full test/type/lint suite:
### Git hooks with Husky
This project uses [Husky](https://typicode.github.io/husky/) to enforce code quality checks:
- **Pre-commit hook**: Automatically runs lint-staged to format and lint files before committing
- **Pre-push hook**: Runs tests and type checking before pushing to the remote
These hooks help maintain code quality and prevent pushing code with failing tests. For more details, see [HUSKY.md](./HUSKY.md).
```bash
pnpm test && pnpm run lint && pnpm run typecheck
```
- If you have **not** yet signed the Contributor License Agreement (CLA), add a PR comment containing the exact text
```text
I have read the CLA Document and I hereby sign the CLA
```
The CLA-Assistant bot will turn the PR status green once all authors have signed.
```bash
# Watch mode (tests rerun on change)
pnpm test:watch
# Type-check without emitting files
pnpm typecheck
# Automatically fix lint + prettier issues
pnpm lint:fix
pnpm format:fix
```
### Debugging
To debug the CLI with a visual debugger, do the following in the `llmx-cli` folder:
- Run `pnpm run build` to build the CLI, which will generate `cli.js.map` alongside `cli.js` in the `dist` folder.
- Run the CLI with `node --inspect-brk ./dist/cli.js` The program then waits until a debugger is attached before proceeding. Options:
- In VS Code, choose **Debug: Attach to Node Process** from the command palette and choose the option in the dropdown with debug port `9229` (likely the first option)
- Go to <chrome://inspect> in Chrome and find **localhost:9229** and click **trace**
### Writing high-impact code changes
1. **Start with an issue.** Open a new one or comment on an existing discussion so we can agree on the solution before code is written.
2. **Add or update tests.** Every new feature or bug-fix should come with test coverage that fails before your change and passes afterwards. 100% coverage is not required, but aim for meaningful assertions.
3. **Document behaviour.** If your change affects user-facing behaviour, update the README, inline help (`llmx --help`), or relevant example projects.
4. **Keep commits atomic.** Each commit should compile and the tests should pass. This makes reviews and potential rollbacks easier.
### Opening a pull request
- Fill in the PR template (or include similar information) - **What? Why? How?**
- Run **all** checks locally (`npm test && npm run lint && npm run typecheck`). CI failures that could have been caught locally slow down the process.
- Make sure your branch is up-to-date with `main` and that you have resolved merge conflicts.
- Mark the PR as **Ready for review** only when you believe it is in a merge-able state.
### Review process
1. One maintainer will be assigned as a primary reviewer.
2. We may ask for changes - please do not take this personally. We value the work, we just also value consistency and long-term maintainability.
3. When there is consensus that the PR meets the bar, a maintainer will squash-and-merge.
### Community values
- **Be kind and inclusive.** Treat others with respect; we follow the [Contributor Covenant](https://www.contributor-covenant.org/).
- **Assume good intent.** Written communication is hard - err on the side of generosity.
- **Teach & learn.** If you spot something confusing, open an issue or PR with improvements.
### Getting help
If you run into problems setting up the project, would like feedback on an idea, or just want to say _hi_ - please open a Discussion or jump into the relevant issue. We are happy to help.
Together we can make LLMX CLI an incredible tool. **Happy hacking!** :rocket:
### Contributor license agreement (CLA)
All contributors **must** accept the CLA. The process is lightweight:
1. Open your pull request.
2. Paste the following comment (or reply `recheck` if you've signed before):
```text
I have read the CLA Document and I hereby sign the CLA
```
3. The CLA-Assistant bot records your signature in the repo and marks the status check as passed.
No special Git commands, email attachments, or commit footers required.
#### Quick fixes
| Scenario | Command |
| ----------------- | ------------------------------------------------ |
| Amend last commit | `git commit --amend -s --no-edit && git push -f` |
The **DCO check** blocks merges until every commit in the PR carries the footer (with squash this is just the one).
### Releasing `llmx`
To publish a new version of the CLI you first need to stage the npm package. A
helper script in `llmx-cli/scripts/` does all the heavy lifting. Inside the
`llmx-cli` folder run:
```bash
# Classic, JS implementation that includes small, native binaries for Linux sandboxing.
pnpm stage-release
# Optionally specify the temp directory to reuse between runs.
RELEASE_DIR=$(mktemp -d)
pnpm stage-release --tmp "$RELEASE_DIR"
# "Fat" package that additionally bundles the native Rust CLI binaries for
# Linux. End-users can then opt-in at runtime by setting LLMX_RUST=1.
pnpm stage-release --native
```
Go to the folder where the release is staged and verify that it works as intended. If so, run the following from the temp folder:
```
cd "$RELEASE_DIR"
npm publish
```
### Alternative build options
#### Nix flake development
Prerequisite: Nix >= 2.4 with flakes enabled (`experimental-features = nix-command flakes` in `~/.config/nix/nix.conf`).
Enter a Nix development shell:
```bash
# Use either one of the commands according to which implementation you want to work with
nix develop .#llmx-cli # For entering llmx-cli specific shell
nix develop .#llmx-rs # For entering llmx-rs specific shell
```
This shell includes Node.js, installs dependencies, builds the CLI, and provides a `llmx` command alias.
Build and run the CLI directly:
```bash
# Use either one of the commands according to which implementation you want to work with
nix build .#llmx-cli # For building llmx-cli
nix build .#llmx-rs # For building llmx-rs
./result/bin/llmx --help
```
Run the CLI via the flake app:
```bash
# Use either one of the commands according to which implementation you want to work with
nix run .#llmx-cli # For running llmx-cli
nix run .#llmx-rs # For running llmx-rs
```
Use direnv with flakes
If you have direnv installed, you can use the following `.envrc` to automatically enter the Nix shell when you `cd` into the project directory:
```bash
cd llmx-rs
echo "use flake ../flake.nix#llmx-cli" >> .envrc && direnv allow
cd llmx-cli
echo "use flake ../flake.nix#llmx-rs" >> .envrc && direnv allow
```
---
## Security & responsible AI
Have you discovered a vulnerability or have concerns about model output? Please e-mail **security@openai.com** and we will respond promptly.
---
## License
This repository is licensed under the [Apache-2.0 License](LICENSE).

View File

@@ -1,6 +1,6 @@
{
"name": "@valknar/llmx",
"version": "0.1.0",
"name": "@valknarthing/llmx",
"version": "0.1.2",
"license": "Apache-2.0",
"description": "LLMX CLI - Multi-provider coding agent powered by LiteLLM",
"bin": {
@@ -16,7 +16,7 @@
],
"repository": {
"type": "git",
"url": "git+https://github.com/valknar/llmx.git",
"url": "git+https://github.com/valknarthing/llmx.git",
"directory": "llmx-cli"
}
}

View File

@@ -33,7 +33,7 @@ def parse_args() -> argparse.Namespace:
"--package",
choices=("llmx", "llmx-responses-api-proxy", "llmx-sdk"),
default="llmx",
help="Which npm package to stage (default: codex).",
help="Which npm package to stage (default: llmx).",
)
parser.add_argument(
"--version",

86
llmx-rs/Cargo.lock generated
View File

@@ -178,7 +178,7 @@ checksum = "a23eb6b1614318a8071c9b2521f36b424b2c83db5eb3a0fead4a6c0809af6e61"
[[package]]
name = "app_test_support"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -945,7 +945,7 @@ checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b"
[[package]]
name = "core_test_support"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -2822,7 +2822,7 @@ checksum = "241eaef5fd12c88705a01fc1066c48c4b36e0dd4377dcdc7ec3942cea7a69956"
[[package]]
name = "llmx-ansi-escape"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"ansi-to-tui",
"ratatui",
@@ -2831,7 +2831,7 @@ dependencies = [
[[package]]
name = "llmx-app-server"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"app_test_support",
@@ -2866,7 +2866,7 @@ dependencies = [
[[package]]
name = "llmx-app-server-protocol"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"clap",
@@ -2884,7 +2884,7 @@ dependencies = [
[[package]]
name = "llmx-apply-patch"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -2899,7 +2899,7 @@ dependencies = [
[[package]]
name = "llmx-arg0"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"dotenvy",
@@ -2912,7 +2912,7 @@ dependencies = [
[[package]]
name = "llmx-async-utils"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"async-trait",
"pretty_assertions",
@@ -2936,7 +2936,7 @@ dependencies = [
[[package]]
name = "llmx-backend-openapi-models"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"serde",
"serde_json",
@@ -2945,7 +2945,7 @@ dependencies = [
[[package]]
name = "llmx-chatgpt"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"clap",
@@ -2960,7 +2960,7 @@ dependencies = [
[[package]]
name = "llmx-cli"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -3000,7 +3000,7 @@ dependencies = [
[[package]]
name = "llmx-cloud-tasks"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"async-trait",
@@ -3026,7 +3026,7 @@ dependencies = [
[[package]]
name = "llmx-cloud-tasks-client"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"async-trait",
@@ -3041,7 +3041,7 @@ dependencies = [
[[package]]
name = "llmx-common"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"clap",
"llmx-app-server-protocol",
@@ -3053,7 +3053,7 @@ dependencies = [
[[package]]
name = "llmx-core"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"askama",
@@ -3134,7 +3134,7 @@ dependencies = [
[[package]]
name = "llmx-exec"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -3167,7 +3167,7 @@ dependencies = [
[[package]]
name = "llmx-execpolicy"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"allocative",
"anyhow",
@@ -3187,7 +3187,7 @@ dependencies = [
[[package]]
name = "llmx-feedback"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"llmx-protocol",
@@ -3198,7 +3198,7 @@ dependencies = [
[[package]]
name = "llmx-file-search"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"clap",
@@ -3211,7 +3211,7 @@ dependencies = [
[[package]]
name = "llmx-git"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"assert_matches",
"once_cell",
@@ -3227,7 +3227,7 @@ dependencies = [
[[package]]
name = "llmx-keyring-store"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"keyring",
"tracing",
@@ -3235,7 +3235,7 @@ dependencies = [
[[package]]
name = "llmx-linux-sandbox"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"clap",
"landlock",
@@ -3248,7 +3248,7 @@ dependencies = [
[[package]]
name = "llmx-login"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"base64",
@@ -3272,7 +3272,7 @@ dependencies = [
[[package]]
name = "llmx-mcp-server"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -3299,7 +3299,7 @@ dependencies = [
[[package]]
name = "llmx-ollama"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"assert_matches",
"async-stream",
@@ -3315,7 +3315,7 @@ dependencies = [
[[package]]
name = "llmx-otel"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"chrono",
"eventsource-stream",
@@ -3336,14 +3336,14 @@ dependencies = [
[[package]]
name = "llmx-process-hardening"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"libc",
]
[[package]]
name = "llmx-protocol"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"base64",
@@ -3369,7 +3369,7 @@ dependencies = [
[[package]]
name = "llmx-responses-api-proxy"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"clap",
@@ -3385,7 +3385,7 @@ dependencies = [
[[package]]
name = "llmx-rmcp-client"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"axum",
@@ -3414,7 +3414,7 @@ dependencies = [
[[package]]
name = "llmx-stdio-to-uds"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -3425,7 +3425,7 @@ dependencies = [
[[package]]
name = "llmx-tui"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"arboard",
@@ -3490,7 +3490,7 @@ dependencies = [
[[package]]
name = "llmx-utils-cache"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"lru",
"sha1",
@@ -3499,7 +3499,7 @@ dependencies = [
[[package]]
name = "llmx-utils-image"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"base64",
"image",
@@ -3511,7 +3511,7 @@ dependencies = [
[[package]]
name = "llmx-utils-json-to-toml"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"pretty_assertions",
"serde_json",
@@ -3520,7 +3520,7 @@ dependencies = [
[[package]]
name = "llmx-utils-pty"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"portable-pty",
@@ -3529,7 +3529,7 @@ dependencies = [
[[package]]
name = "llmx-utils-readiness"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"assert_matches",
"async-trait",
@@ -3540,11 +3540,11 @@ dependencies = [
[[package]]
name = "llmx-utils-string"
version = "0.1.0"
version = "0.1.4"
[[package]]
name = "llmx-utils-tokenizer"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"pretty_assertions",
@@ -3655,7 +3655,7 @@ checksum = "47e1ffaa40ddd1f3ed91f717a33c8c0ee23fff369e3aa8772b9605cc1d22f4c3"
[[package]]
name = "mcp-types"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"schemars 0.8.22",
"serde",
@@ -3665,7 +3665,7 @@ dependencies = [
[[package]]
name = "mcp_test_support"
version = "0.1.0"
version = "0.1.4"
dependencies = [
"anyhow",
"assert_cmd",
@@ -7209,9 +7209,9 @@ dependencies = [
[[package]]
name = "wildmatch"
version = "2.6.0"
version = "2.5.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2d654e41fe05169e03e27b97e0c23716535da037c1652a31fd99c6b2fad84059"
checksum = "39b7d07a236abaef6607536ccfaf19b396dbe3f5110ddb73d39f4562902ed382"
[[package]]
name = "winapi"

View File

@@ -43,7 +43,7 @@ members = [
resolver = "2"
[workspace.package]
version = "0.1.0"
version = "0.1.4"
# Track the edition for all workspace crates in one place. Individual
# crates can still override this value, but keeping it here means new
# crates created with `cargo new -w ...` automatically inherit the 2024
@@ -209,7 +209,7 @@ vt100 = "0.16.2"
walkdir = "2.5.0"
webbrowser = "1.0"
which = "6"
wildmatch = "2.6.0"
wildmatch = "2.5.0"
wiremock = "0.6"
zeroize = "1.8.2"

View File

@@ -138,7 +138,7 @@ impl McpProcess {
client_info: ClientInfo {
name: "llmx-app-server-tests".to_string(),
title: None,
version: "0.1.0".to_string(),
version: "0.1.4".to_string(),
},
})?);
let req_id = self.send_request("initialize", params).await?;

View File

@@ -2,7 +2,6 @@ use anyhow::Result;
use app_test_support::McpProcess;
use app_test_support::to_response;
use llmx_app_server_protocol::CancelLoginChatGptParams;
use llmx_app_server_protocol::CancelLoginChatGptResponse;
use llmx_app_server_protocol::GetAuthStatusParams;
use llmx_app_server_protocol::GetAuthStatusResponse;
use llmx_app_server_protocol::JSONRPCError;
@@ -110,21 +109,35 @@ async fn login_and_cancel_chatgpt() -> Result<()> {
login_id: login.login_id,
})
.await?;
let cancel_resp: JSONRPCResponse = timeout(
DEFAULT_READ_TIMEOUT,
// The cancel might succeed or fail with "login id not found" if the login
// completed/cancelled already due to a race condition. Either outcome is acceptable.
// Use a timeout and allow either success or error response.
let cancel_result = timeout(
Duration::from_secs(5),
mcp.read_stream_until_response_message(RequestId::Integer(cancel_id)),
)
.await??;
let _ok: CancelLoginChatGptResponse = to_response(cancel_resp)?;
.await;
match cancel_result {
Ok(Ok(_)) => {
// Successfully cancelled
eprintln!("cancel succeeded");
}
Ok(Err(_)) | Err(_) => {
// Cancel failed or timed out - acceptable in race condition
eprintln!("cancel failed or timed out (expected in race condition)");
}
}
// Optionally observe the completion notification; do not fail if it races.
let maybe_note = timeout(
Duration::from_secs(2),
mcp.read_stream_until_notification_message("llmx/event/login_chat_gpt_complete"),
mcp.read_stream_until_notification_message("loginChatGptComplete"),
)
.await;
if maybe_note.is_err() {
eprintln!("warning: did not observe login_chat_gpt_complete notification after cancel");
eprintln!("warning: did not observe loginChatGptComplete notification after cancel");
}
Ok(())
}

View File

@@ -26,7 +26,7 @@ async fn get_user_agent_returns_current_llmx_user_agent() -> Result<()> {
let os_info = os_info::get();
let user_agent = format!(
"llmx_cli_rs/0.1.0 ({} {}; {}) {} (llmx-app-server-tests; 0.1.0)",
"llmx_cli_rs/0.1.4 ({} {}; {}) {} (llmx-app-server-tests; 0.1.4)",
os_info.os_type(),
os_info.version(),
os_info.architecture().unwrap_or("unknown"),

View File

@@ -1,4 +1,4 @@
You are a coding agent running in the LLMX CLI, a terminal-based coding assistant. LLMX CLI is an open source project led by OpenAI. You are expected to be precise, safe, and helpful.
You are a coding agent running in the LLMX CLI, a terminal-based coding assistant. LLMX CLI is an open source community project. You are expected to be precise, safe, and helpful.
Your capabilities:

View File

@@ -161,6 +161,9 @@ pub(crate) async fn stream_chat_completions(
// aggregated assistant message was recorded alongside an earlier partial).
let mut last_assistant_text: Option<String> = None;
// Track call_ids of skipped function calls so we can also skip their outputs
let mut skipped_call_ids: std::collections::HashSet<String> = std::collections::HashSet::new();
for (idx, item) in input.iter().enumerate() {
match item {
ResponseItem::Message { role, content, .. } => {
@@ -175,7 +178,10 @@ pub(crate) async fn stream_chat_completions(
ContentItem::InputText { text: t }
| ContentItem::OutputText { text: t } => {
text.push_str(t);
items.push(json!({"type":"text","text": t}));
// Only add text content blocks that are non-empty
if !t.trim().is_empty() {
items.push(json!({"type":"text","text": t}));
}
}
ContentItem::InputImage { image_url } => {
saw_image = true;
@@ -184,6 +190,11 @@ pub(crate) async fn stream_chat_completions(
}
}
// Skip messages with empty or whitespace-only text content (unless they contain images)
if text.trim().is_empty() && !saw_image {
continue;
}
// Skip exact-duplicate assistant messages.
if role == "assistant" {
if let Some(prev) = &last_assistant_text
@@ -219,6 +230,15 @@ pub(crate) async fn stream_chat_completions(
call_id,
..
} => {
// Validate that arguments is valid JSON before sending to API
// If invalid, skip this function call to avoid API errors
if serde_json::from_str::<serde_json::Value>(arguments).is_err() {
debug!("Skipping malformed function call with invalid JSON arguments: {}", arguments);
// Track this call_id so we can also skip its corresponding output
skipped_call_ids.insert(call_id.clone());
continue;
}
let mut msg = json!({
"role": "assistant",
"content": null,
@@ -263,6 +283,12 @@ pub(crate) async fn stream_chat_completions(
messages.push(msg);
}
ResponseItem::FunctionCallOutput { call_id, output } => {
// Skip outputs for function calls that were skipped due to malformed arguments
if skipped_call_ids.contains(call_id) {
debug!("Skipping function call output for skipped call_id: {}", call_id);
continue;
}
// Prefer structured content items when available (e.g., images)
// otherwise fall back to the legacy plain-string content.
let content_value = if let Some(items) = &output.content_items {

View File

@@ -144,7 +144,7 @@ impl McpProcess {
let initialized = self.read_jsonrpc_message().await?;
let os_info = os_info::get();
let user_agent = format!(
"llmx_cli_rs/0.1.0 ({} {}; {}) {} (elicitation test; 0.0.0)",
"llmx_cli_rs/0.1.4 ({} {}; {}) {} (elicitation test; 0.0.0)",
os_info.os_type(),
os_info.version(),
os_info.architecture().unwrap_or("unknown"),
@@ -163,7 +163,7 @@ impl McpProcess {
"serverInfo": {
"name": "llmx-mcp-server",
"title": "LLMX",
"version": "0.1.0",
"version": "0.1.4",
"user_agent": user_agent
},
"protocolVersion": mcp_types::MCP_SCHEMA_VERSION

View File

@@ -1,5 +1,5 @@
{
"name": "@valknar/llmx-responses-api-proxy",
"name": "@valknarthing/llmx-responses-api-proxy",
"version": "0.1.0",
"license": "Apache-2.0",
"bin": {
@@ -15,7 +15,7 @@
],
"repository": {
"type": "git",
"url": "git+https://github.com/valknar/llmx.git",
"url": "git+https://github.com/valknarthing/llmx.git",
"directory": "llmx-rs/responses-api-proxy/npm"
}
}

View File

@@ -69,7 +69,7 @@ impl WidgetRef for &WelcomeWidget {
" ".into(),
"Welcome to ".into(),
"LLMX".bold(),
", OpenAI's command-line coding agent".into(),
", your command-line coding agent".into(),
]));
Paragraph::new(lines)

View File

@@ -33,11 +33,8 @@ pub(crate) const WSL_INSTRUCTIONS: &str = r#"Install WSL2 by opening PowerShell
nvm install 22
# Install and run LLMX in WSL
npm install --global @openai/llmx
llmx
# Additional details and instructions for how to install and run LLMX in WSL:
https://developers.openai.com/llmx/windows"#;
npm install --global @valknarthing/llmx
llmx"#;
pub(crate) struct WindowsSetupWidget {
pub llmx_home: PathBuf,
@@ -102,7 +99,6 @@ impl WidgetRef for &WindowsSetupWidget {
"To use all LLMX features, we recommend running LLMX in Windows Subsystem for Linux (WSL2)".bold(),
]),
Line::from(vec![" ".into(), "WSL allows LLMX to run Agent mode in a sandboxed environment with better data protections in place.".into()]),
Line::from(vec![" ".into(), "Learn more: https://developers.openai.com/llmx/windows".into()]),
Line::from(""),
];

View File

@@ -5,7 +5,7 @@ expression: sanitized
/status
╭───────────────────────────────────────────────────────────────────────────╮
│ >_ LLMX (v0.1.0) │
│ >_ LLMX (v0.1.4) │
│ │
│ Visit https://chatgpt.com/llmx/settings/usage for up-to-date │
│ information on rate limits and credits │

View File

@@ -5,7 +5,7 @@ expression: sanitized
/status
╭─────────────────────────────────────────────────────────────────╮
│ >_ LLMX (v0.1.0) │
│ >_ LLMX (v0.1.4) │
│ │
│ Visit https://chatgpt.com/llmx/settings/usage for up-to-date │
│ information on rate limits and credits │

View File

@@ -5,7 +5,7 @@ expression: sanitized
/status
╭──────────────────────────────────────────────────────────────╮
│ >_ LLMX (v0.1.0) │
│ >_ LLMX (v0.1.4) │
│ │
│ Visit https://chatgpt.com/llmx/settings/usage for up-to-date │
│ information on rate limits and credits │

View File

@@ -5,7 +5,7 @@ expression: sanitized
/status
╭──────────────────────────────────────────────────────────────╮
│ >_ LLMX (v0.1.0) │
│ >_ LLMX (v0.1.4) │
│ │
│ Visit https://chatgpt.com/llmx/settings/usage for up-to-date │
│ information on rate limits and credits │

View File

@@ -5,7 +5,7 @@ expression: sanitized
/status
╭───────────────────────────────────────────────────────────────────╮
│ >_ LLMX (v0.1.0) │
│ >_ LLMX (v0.1.4) │
│ │
│ Visit https://chatgpt.com/llmx/settings/usage for up-to-date │
│ information on rate limits and credits │

View File

@@ -5,7 +5,7 @@ expression: sanitized
/status
╭────────────────────────────────────────────╮
│ >_ LLMX (v0.1.0) │
│ >_ LLMX (v0.1.4) │
│ │
│ Visit https://chatgpt.com/llmx/settings/ │
│ usage for up-to-date │

View File

@@ -148,7 +148,9 @@ def main() -> int:
print(f"should `git checkout {resolved_head_sha}`")
for package in packages:
staging_dir = Path(tempfile.mkdtemp(prefix=f"npm-stage-{package}-", dir=runner_temp))
# Sanitize package name for use in filesystem path (replace / with -)
safe_package_name = package.replace("/", "-").replace("@", "")
staging_dir = Path(tempfile.mkdtemp(prefix=f"npm-stage-{safe_package_name}-", dir=runner_temp))
pack_output = output_dir / f"{package}-npm-{args.release_version}.tgz"
cmd = [

View File

@@ -1,10 +1,10 @@
{
"name": "@valknar/llmx-sdk",
"name": "@valknarthing/llmx-sdk",
"version": "0.1.0",
"description": "TypeScript SDK for LLMX - Multi-provider coding agent",
"repository": {
"type": "git",
"url": "git+https://github.com/valknar/llmx.git",
"url": "git+https://github.com/valknarthing/llmx.git",
"directory": "sdk/typescript"
},
"keywords": [