Run VirtuAI deep agents on your local machine
Project description
VirtuAI CLI
Run VirtuAI agents from your terminal — chat interactively in a Claude-Code-style TUI, or pipe them into shell scripts. The agent's bash/file tools run on your machine, so files appear in your filesystem and commands hit your real shell.
Installation
pip install virtuai-cli
Or with pipx (recommended for CLI tools — handles PATH automatically):
pipx install virtuai-cli
Requires Python 3.11+.
Quick start
1. Pair this machine with a workspace
Open the VirtuAI portal → Settings → CLI → generate a pairing code, then:
virtuai pair <CODE>
The pairing is one-time; the resulting token is stored in your system keychain.
2. Chat with an agent
virtuai chat
Opens an interactive TUI. Pick an agent (skipped if your workspace has only one), type a message, watch the response stream in. The agent's tool calls — bash, file reads, file writes — execute on this machine, in the current directory.
3. Use it from a shell script
virtuai ask "summarize the changes in this branch"
git diff | virtuai ask "what does this change?"
virtuai ask --json "find any bugs" > events.jsonl
ask is non-interactive: it prints the response to stdout, then exits.
Commands
| Command | Description |
|---|---|
virtuai chat |
Open the interactive chat TUI |
virtuai ask "..." |
One-shot prompt for scripts (reads stdin if piped) |
virtuai pair <code> |
Pair this machine with a workspace |
virtuai unpair |
Revoke the current pairing |
virtuai run |
Headless WebSocket runner — let the web app/channels invoke local tools without opening the TUI |
virtuai status |
Show pairing and connection status |
virtuai logs |
Show recent commands the agent executed locally |
virtuai login |
Authenticate with your VirtuAI account (browser flow) |
virtuai config get/set |
Read or set local CLI options (e.g. server_url) |
Inside virtuai chat
The TUI streams tokens, renders tool calls as live cards (running → complete), tracks the agent's todo list, and shows a randomized "✻ Pondering…" indicator while the agent is composing.
Slash commands (type / to see the filtered list, Tab to autocomplete):
| Slash command | What it does |
|---|---|
/help |
Show this list |
/clear, /new |
Drop the current session and start fresh |
/history |
List your recent conversations with this agent |
/load <id> |
Reopen a past conversation by session_id |
/models |
List models available for this agent |
/model <id> |
Switch the model for subsequent messages |
/exit, /quit |
Close the TUI |
Key bindings:
- Esc — cancel the in-flight response
- Ctrl+L — start a new conversation
- Ctrl+C — quit
- Tab — autocomplete the current
/command
virtuai ask for scripting
# Direct prompt
virtuai ask "what's in this folder?"
# Pipe stdin in (concatenated with the argument if both given)
git log --oneline -20 | virtuai ask "summarize these commits"
# Continue a previous session (use --print-session to surface the id)
virtuai ask --session sess-aaa "and apply that fix" --print-session
# Machine-readable: every event as one JSON line
virtuai ask --json "find any bugs" > events.jsonl
# Just the final answer, no streaming (good for capturing into variables)
ANSWER=$(virtuai ask -q "give me the current time")
# Pick a different agent or model
virtuai ask --agent code-review --model claude-opus "review main.py"
# Skip the local runner for a faster startup when no tools are needed
virtuai ask --no-tools "tell me a joke"
Output streams:
stream(default) — tokens go to stdout, tool indicators to stderr (so> out.txtcaptures only the answer)--quiet/-q— accumulate, print only the final assistant text at the end--json— every SSE event as one JSONL line on stdout (parseable)
Exit codes: 0 success · 1 stream error · 2 bad args / agent not found.
Common options
--server <url> Override the saved server URL
--workdir <path> Working directory for tool execution (default: current dir)
--agent <id|name> Pick a specific agent
--model <id> Override the agent's default model
Security model
Read this section before pointing an agent at anything you care about.
The CLI runs the agent's shell commands as your user account, with your full filesystem permissions. There is no sandbox. The agent can do anything you can do at the shell. The guardrails below are best described as speed bumps, not walls.
What is enforced
- Auth: CLI token is stored in your OS keychain (Keychain on macOS, libsecret on Linux, Credential Manager on Windows). All traffic is TLS (WSS + HTTPS) with the
certifiCA bundle. - File-tool path jail: the deepagents file tools —
write_file,edit_file,read_file— route throughupload_files/download_files, which reject absolute paths outside the workdir. This part works. cdjail in bash: the executor wraps every shell command in a bash subshell with a customcdfunction that refuses to leave the workdir. Catches the trivialcd ..case.- Denylist: a regex match on the command string blocks
sudo,rm -rf /, fork bombs,mkfs,dd of=/dev/.... Easy to bypass with aliasing or encoding; mostly there to stop drive-by accidents. - Audit log: every executed command + exit code goes to
~/.virtuai/audit.logwith a timestamp. - Foreground-only: the agent only has access while
virtuai chat,virtuai ask, orvirtuai runis running — no background daemon.
What is NOT enforced
- Shell commands can use absolute paths anywhere. The
cdjail only interceptscd.cat /etc/passwd,ls /Users/you/Documents,rm /tmp/whatever— none of these touchcd, so none are blocked. The agent can read, write, and delete anything you can. - The
cdjail itself is bash-only.sh -c '...',python -c "os.chdir('/')",node -e "..."all bypass it. - Command substitution, eval, env vars, aliases all defeat the denylist.
$(echo s)$(echo udo) ...looks nothing likesudoto a regex. - Network egress is not restricted. The agent can
curl/wgetanywhere, exfiltrate to any host.
If you need real isolation
The only honest options:
- Run the CLI inside a container with the workdir bind-mounted and nothing else:
docker run --rm -it -v "$PWD:/work" -w /work python:3.12 bash -c "pip install virtuai-cli && virtuai pair <code> && virtuai chat"
The container has access only to what you mount. - OS-level sandboxing: wrap
virtuai runinsandbox-execon macOS or run inside alandlock/bubblewrapjail on Linux. - Switch the agent's backend to the E2B remote sandbox in the portal. The agent then runs against a disposable VM, not your laptop. The CLI runner becomes optional (you'd only need it for "interactive" workflows where local files matter).
Use the CLI runner the same way you'd use a powerful shell session you don't fully trust. Point it at a project directory, never at ~, and don't run unattended on a machine with credentials lying around.
Programmatic use
The async building blocks the TUI is built on are importable directly:
import asyncio
from virtuai_cli import config as cfg
from virtuai_cli.chat.sse import stream_chat
async def main():
token = cfg.load_token(cfg._KEYRING_CLI_TOKEN_KEY)
server = cfg.get_server_url()
async for event in stream_chat(server, token, "your-agent-id", "Hello"):
if event.get("type") == "token":
print(event["content"], end="", flush=True)
asyncio.run(main())
A formal SDK (typed client class, sync wrappers) is on the roadmap.
Links
- Homepage: https://imvirtuai.com
- PyPI: https://pypi.org/project/virtuai-cli/
- Issues / docs: contact your workspace admin
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file virtuai_cli-0.8.0.tar.gz.
File metadata
- Download URL: virtuai_cli-0.8.0.tar.gz
- Upload date:
- Size: 37.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.11
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
406c28e60051c2a0a92431867058b6f487b89a130e2010796fb298ae71e77ca2
|
|
| MD5 |
d03ff8c128324b363c0b8db368248d89
|
|
| BLAKE2b-256 |
1cc6077de6ec22cfa6c917b71dd604f5e9f61c4919d39a9d77f4e71b53349f15
|
File details
Details for the file virtuai_cli-0.8.0-py3-none-any.whl.
File metadata
- Download URL: virtuai_cli-0.8.0-py3-none-any.whl
- Upload date:
- Size: 39.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.11
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3c2f2062d818d03b0a5756be7fec3f4294e2d62320091cad78f84949366b9a32
|
|
| MD5 |
411a96459722ed8f48a0533ae246fdeb
|
|
| BLAKE2b-256 |
720f3efcbc8b34818ec556ae892c9568b21e0b919a240e34457f16e5aac6b7be
|