Skip to main content

Call the Google Antigravity CLI (agy) headlessly from any non-TTY context (subprocess, MCP, CI). Works around upstream bug #76.

Project description

agy-headless-bridge

Call the Google Antigravity CLI (agy) headlessly โ€” and actually get output back.

Codename PtyGravity ยท pty + antiGravity

PyPI PyPI downloads MCP Registry tests License: MIT Python Platform

๐Ÿ“– Architecture & docs โ†’ rhishi99.github.io/agy-headless-bridge


TL;DR โ€” the problem, before & after

agy -p "<prompt>" prints nothing when its stdout is not a real terminal. So calling it from a subprocess, an MCP server, CI, or another coding agent (Claude Code, Codex, โ€ฆ) returns an empty string and exit 0 โ€” silently. This package gives agy a fresh pseudo-terminal so it emits normally, then cleans the output.

flowchart TB
    subgraph B["โŒ BEFORE โ€” agy -p from any non-TTY caller"]
        direction TB
        a1["subprocess ยท MCP ยท CI ยท agent"] --> a2["agy -p &quot;prompt&quot;"]
        a2 --> a3["stdout gated by isatty()"]
        a3 --> a4["(empty string)<br/>exit 0 ยท no error ยท no output"]
    end
    subgraph A["โœ… AFTER โ€” through agy-headless-bridge"]
        direction TB
        b1["subprocess ยท MCP ยท CI ยท agent"] --> b2["run(prompt)"]
        b2 --> b3["allocate fresh pseudo-terminal"]
        b3 --> b4["agy -p &quot;prompt&quot;<br/>isatty() == True"]
        b4 --> b5["clean() strips ANSI/TUI"]
        b5 --> b6["clean text โœ“"]
    end

    classDef bad fill:#2a1313,stroke:#f87171,color:#ffd9d9;
    classDef good fill:#0f2a1e,stroke:#34d399,color:#d7ffe9;
    class a1,a2,a3,a4 bad;
    class b1,b2,b3,b4,b5,b6 good;
# โŒ The problem โ€” plain subprocess
import subprocess
r = subprocess.run(["agy", "-p", "say hi"], capture_output=True, text=True)
print(r.stdout)          # '' โ€” prints nothing, exit 0

# โœ… The fix
from agy_headless_bridge import run
print(run("say hi"))     # 'Hi! How can I help?'

Three entry points around one core:

Entry point Invoke Use for
Library from agy_headless_bridge import run embedding agy in Python
CLI agy-bridge "prompt" shell scripts, quick calls
MCP server python -m agy_headless_bridge.mcp_server letting an agent call agy as a tool

The problem in detail โ€” upstream bug #76

agy gates its stdout on isatty(). The instant stdout isn't a terminal, it goes silent โ€” no output, no error, exit 0:

$ agy -p "say hi" | cat
$            # empty. exit 0. nothing.

The common winpty agy -p "..." workaround needs a terminal that already exists, so it still fails from any automated / non-TTY caller.

The fix โ€” give agy a tty it didn't ask for

Allocate a brand-new pseudo-terminal (one that needs no parent tty) and attach agy to it. Same code path on every OS โ€” only the pty allocator differs.

flowchart TD
    A["Caller โ€” non-TTY<br/>Claude Code ยท MCP ยท subprocess ยท CI"] -->|"prompt"| B{{"run(prompt)"}}
    B --> C["find_agy()<br/>$AGY_PATH โ†’ PATH โ†’ OS defaults"]
    C --> D{"sys.platform?"}
    D -->|"win32"| E["pywinpty<br/>PtyProcess.spawn"]
    D -->|"posix"| F["stdlib pty<br/>os.openpty + Popen"]
    E --> G(["fresh pseudo-terminal"])
    F --> G
    G --> H["agy -p prompt<br/>isatty == True โ†’ emits"]
    H -->|"raw bytes + ANSI/TUI chrome"| I["clean()<br/>strip CSI/OSC ยท collapse \r repaints ยท drop spinner glyphs"]
    I -->|"clean text"| A
Platform pty backend Status
Windows ConPTY via pywinpty (PtyProcess) โœ… verified (agy 1.0.6)
Linux / macOS stdlib pty (os.openpty + subprocess.Popen) โš ๏ธ pty mechanics verified on Linux CI; real agy round-trip not yet verified on POSIX hardware

[!TIP] Linux/macOS users wanted. The pty mechanics (the os.openpty + Popen plumbing) are verified on Linux CI via a stub, but nobody has confirmed the real agy round-trip on bare-metal Linux or macOS yet. If you're on POSIX: pip install agy-headless-bridge, try it, and tell us how it went โ€” pass or fail. PRs welcome.

Why not just the existing agy Claude Code plugins? They wrap agy for triggering (slash commands, model selection) but still call agy -p directly โ€” so in any headless context they hit this exact empty-output bug. This package fixes the I/O layer they're missing. Use both together.


Prerequisites

Before installing this bridge you need:

  1. Python 3.9+ โ€” python --version.
  2. The Antigravity CLI (agy), installed and authenticated:
    • Install: https://antigravity.google/cli
    • Authenticate once interactively (agy opens a browser OAuth flow), or set ANTIGRAVITY_API_KEY in your environment if you use an API key.
    • Verify it runs in a real terminal: agy -p "say hi" should print a reply. (From a pipe it won't โ€” that's the very bug this package fixes.)
  3. Windows only: pywinpty (installed automatically as a dependency). POSIX uses the stdlib pty module โ€” nothing extra.

This package does not install or authenticate agy, and does not bundle any credentials. It only spawns the agy already on your machine.


Install

Requires Python 3.9+.

pip install agy-headless-bridge          # pywinpty auto-installs on Windows only

From source:

git clone https://github.com/rhishi99/agy-headless-bridge
cd agy-headless-bridge
pip install -e .

The bridge locates the binary via, in order: $AGY_PATH โ†’ agy on PATH โ†’ OS default install paths.


Usage

Library

from agy_headless_bridge import run, AgyNotFoundError, AgyTimeoutError

try:
    # For a CODING task, pass the repo so agy can see it โ€” the library `run()`
    # is a thin primitive and does NOT auto-add a workspace (the CLI/MCP layers
    # do). Without add_dirs, agy -p runs blind in its own scratch workspace.
    print(run("Summarize this repo", add_dirs=["."], timeout=600))
except AgyTimeoutError as exc:
    print("timed out; partial so far:\n", exc.partial)  # resume with `agy -c`
except AgyNotFoundError:
    print("install agy first")

run(prompt, timeout=900, agy_path=None, *, add_dirs=None, model=None, idle_timeout=120, extra_args=None) -> str โ€” raises AgyNotFoundError if the binary is missing, AgyTimeoutError (carrying .partial) on the idle or hard timeout, ValueError on empty prompt. Returns "" only if agy genuinely emitted nothing. add_dirs โ†’ agy --add-dir; model โ†’ --model. The timeout is the absolute ceiling; idle_timeout ends a run that has gone silent for that many seconds (so a long-but-active task survives the ceiling).

To get the CLI/MCP "default to cwd for coding, none for research" behaviour in your own code, use resolve_add_dirs(explicit, use_cwd_default=...).

CLI

agy-bridge "reply with exactly: OK"
python -m agy_headless_bridge "reply with exactly: OK"   # equivalent

# Coding task: the current directory is auto-added to agy's workspace, so agy
# sees your repo with no extra flags.
agy-bridge "Find and fix the off-by-one in the parser"

# Research / Q&A that needs no repo โ€” opt out of the cwd default:
agy-bridge --no-workspace "Explain idle timeouts in process supervision"

# Add more dirs, pick a model, widen the limits for a big task:
agy-bridge --add-dir ../shared-lib --model gemini-3-pro \
    --timeout 1800 --idle-timeout 180 "Refactor X to match the shared lib"
Flag Default Meaning
--add-dir DIR cwd (auto) Add a dir to agy's workspace (repeatable).
--no-workspace off Don't auto-add cwd (use for research / Q&A).
--model NAME agy default agy --model.
--timeout SECS 900 Hard ceiling (absolute wall).
--idle-timeout SECS 120 Kill after this many seconds of no output.

MCP server

claude mcp add --transport stdio antigravity -- agy-mcp-server

agy-mcp-server is the console-script entry point pip install puts on your PATH โ€” it's bound to the exact interpreter you installed the package with, so it sidesteps the "wrong python" problem below entirely. Prefer it over python -m agy_headless_bridge.mcp_server, especially on Windows.

Alternative: invoke via python -m
claude mcp add --transport stdio antigravity -- \
    python -m agy_headless_bridge.mcp_server

Windows: use py -3.11 (the Python Launcher) instead of python in the command above. Bare python can resolve to the wrong interpreter or the Windows Store stub, which surfaces as an -32000 MCP connection error. See Troubleshooting.

The server speaks JSON-RPC stdio directly (no MCP SDK dependency) and routes every call through the pty bridge.

Tool schema (what an agent โ€” or you, integrating manually โ€” sees):

Tool Argument Type Required Description
agy_ask prompt string โœ… one-shot prompt sent to agy
agy_ask workspace "auto"|"none" auto (default) adds the server's cwd so agy sees the repo; none for research / Q&A
agy_ask add_dir string[] explicit dirs for agy's workspace (overrides the workspace default)
agy_ask model string agy --model
agy_ask timeout number hard timeout override, seconds
agy_research query string โœ… wrapped as a deep-research prompt for agy (never attaches a workspace)

Response shape โ€” a standard MCP tools/call result; the answer is the text content:

{
  "jsonrpc": "2.0",
  "id": 2,
  "result": { "content": [ { "type": "text", "text": "<agy's cleaned answer>" } ] }
}

On a timeout the text is whatever agy produced before the kill followed by an [agy-mcp] TIMEOUT: ...; resume with 'agy -c' note, so partial work isn't lost. On other failures the text is an [agy-mcp] ERROR: ... string (agy missing, etc.) rather than a JSON-RPC error, so the agent always gets a readable reply.


Use cases & wiring it into your AI coding tools

The whole point: let one AI coding tool delegate work to Gemini via Antigravity, headlessly. Common setups:

Use case How
Claude Code asks Gemini for a second opinion / diff review MCP server โ†’ agy_ask tool
A CI step runs an agy prompt and captures the answer agy-bridge "..." in the workflow
A Python pipeline fans work out to agy from agy_headless_bridge import run
Codex / any MCP-capable agent delegates to agy register the same MCP server
Cron / scheduled job summarizes logs via agy agy-bridge in the script

Wire into Claude Code

Register the MCP server, then prompt Claude to use it:

claude mcp add --transport stdio antigravity -- agy-mcp-server

Prompt to Claude Code: "Use the agy_ask tool to ask Antigravity to review this function for edge cases, then summarize its findings for me."

If you also want slash-command triggering and model selection, pair this bridge with the community antigravity-cc Claude Code plugin โ€” that handles the /agy:* commands and Gemini/Claude model swap; this handles the headless I/O.

Wire into Codex (or any MCP client)

Add the server to the client's MCP config:

{
  "mcpServers": {
    "antigravity": {
      "command": "python",
      "args": ["-m", "agy_headless_bridge.mcp_server"]
    }
  }
}

Prompt to the agent: "Call agy_research with the query 'idiomatic error handling in Rust' and turn the result into a checklist."

Use from a shell / CI script

ANSWER="$(agy-bridge 'Summarize the key risk in this diff in one sentence.')"
echo "$ANSWER"

Configuration

Env var Default Meaning
AGY_PATH auto-detect Absolute path to the agy binary
AGY_BRIDGE_TIMEOUT 900 Hard ceiling (absolute wall), in seconds
AGY_BRIDGE_IDLE_TIMEOUT 120 Kill after this many seconds of no output from agy

How clean() works

agy's pty output is a TUI stream, not plain text. clean() removes ANSI escapes (CSI/OSC โ€” colors, cursor moves), \r repaints (a spinner overwrites one line; only the final paint is kept), and box-drawing / spinner glyphs (โ•ญโ”€โ•ฎ โ”‚ โ ‹โ ™โ น) โ€” leaving just the model's answer.

What comes off the pty vs. what you get back:

RAW (off the pty)                          CLEANED (returned to you)
โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€     โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
โ ‹ thinkingโ€ฆ\rโ ™ thinkingโ€ฆ\r\x1b[2K          A closure is a function that
\x1b[32mโ•ญโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ•ฎ\x1b[0m              captures variables from the
\x1b[32mโ”‚\x1b[0m A closure is a function     scope where it was defined.
that captures variables from the
scope where it was defined.
\x1b[32mโ•ฐโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ•ฏ\x1b[0m

Troubleshooting / FAQ

pip install fails on Windows building pywinpty โ€” pywinpty is a native extension. If pip tries to build from source and errors with a compiler/cl.exe message, install the Microsoft C++ Build Tools (or use a Python where a prebuilt pywinpty wheel exists โ€” recent CPython on Windows has them). Upgrade pip first: python -m pip install -U pip.

-32000 MCP connection error on Windows (interpreter mismatch) โ€” bare python in the claude mcp add command can resolve to the wrong interpreter or the Windows Store stub, so the spawned server can't import the package and the connection fails. Use the Python Launcher instead โ€” register the server with py -3.11 (matching the Python where you installed the package) in place of python. To confirm where the package actually landed, run pip show agy-headless-bridge and check the Location: field; if it points to a different Python than Claude Code spawns, that mismatch is the cause.

AgyNotFoundError โ€” the bridge can't find agy. Set AGY_PATH to the absolute path of the binary, or make sure agy is on your PATH (agy --version should work in your shell).

Empty string returned โ€” agy produced no output. Confirm it works in a real terminal first: agy -p "say hi". If that's also empty, the problem is agy/auth, not the bridge. Re-authenticate (agy interactively) or check ANTIGRAVITY_API_KEY.

AgyTimeoutError โ€” agy hit a limit. Two triggers: the hard ceiling (AGY_BRIDGE_TIMEOUT, default 900s) and the idle timeout (AGY_BRIDGE_IDLE_TIMEOUT, default 120s โ€” fires when agy emits nothing for that long). The error carries .partial (whatever agy produced first), and the CLI prints it before exiting. Widen the limits for long prompts: AGY_BRIDGE_TIMEOUT=1800 agy-bridge "...", --idle-timeout 300, or run(prompt, timeout=1800, idle_timeout=300). To continue a run that timed out, resume the agy session directly with agy -c.

agy stalls until the idle timeout, then times out on what looks like a normal prompt โ€” this bridge never writes to agy's stdin, so if agy pauses mid-run to ask for interactive approval (e.g. "allow this tool call?"), the prompt sits unanswered and the run silently stalls until the idle timeout kills it. This is currently the most common way a run hangs. Configure agy itself to auto-approve (check agy --help / its settings for a non-interactive/auto-approve flag) before delegating tasks that involve tool use, or expect the idle timeout to eventually fire and surface .partial.

Pseudo-terminal allocation fails โ€” rare. On Windows it means pywinpty isn't importable (reinstall it). On POSIX it means the system is out of pty slots or pty.openpty() is denied (containers with no /dev/pts); run with a real pty available.

Garbled / partial output โ€” open an issue with the OS, Python + agy version, and the raw output; clean() may need another glyph rule.

Development & CI

pip install -e ".[dev]"
pytest

Unit tests (cleaning, arg validation, binary discovery) always run. The live agy round-trip test auto-skips when agy isn't installed โ€” so CI runners (which don't have agy) stay green and never need credentials. CI runs on Windows + Linux across Python 3.9 and 3.12.


Scope, non-goals & disclaimer

  • Model selection (Gemini Pro / Flash / Claude inside agy) is not handled here โ€” it's an agy settings.json concern, covered by the antigravity-cc plugin. Pair the two.
  • Does not install or authenticate agy, and ships no credentials.
  • Automating any vendor CLI may interact with that vendor's terms / rate limits. You are responsible for using agy within Google's terms of service. This project only changes how stdout is captured โ€” it does not bypass auth, quotas, or any access control.
  • Not affiliated with Google. Antigravity and agy are Google products.

License

MIT.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

agy_headless_bridge-1.2.1.tar.gz (29.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

agy_headless_bridge-1.2.1-py3-none-any.whl (20.3 kB view details)

Uploaded Python 3

File details

Details for the file agy_headless_bridge-1.2.1.tar.gz.

File metadata

  • Download URL: agy_headless_bridge-1.2.1.tar.gz
  • Upload date:
  • Size: 29.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for agy_headless_bridge-1.2.1.tar.gz
Algorithm Hash digest
SHA256 eb83f9e453bc84d609e3736de29e8ab7f8cd91d2d6e707ab77c443617e276970
MD5 4321ea7d1853fdbe98df41e75d7107c8
BLAKE2b-256 e10d76f3cce3bb711fde7f7e4640540165d25346be19f611cf7fe850deb43ced

See more details on using hashes here.

Provenance

The following attestation bundles were made for agy_headless_bridge-1.2.1.tar.gz:

Publisher: publish.yml on rhishi99/agy-headless-bridge

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file agy_headless_bridge-1.2.1-py3-none-any.whl.

File metadata

File hashes

Hashes for agy_headless_bridge-1.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 4f3c826a8a9d62984fdd0548cad2f16f38ea645ab01eb4b318803c401bc8022c
MD5 1bf163ebe4bfb29e9cf4b55ff523ffbf
BLAKE2b-256 44be4cd0520a43127e3f8ba3fe0ee0f35028d2dff0f8c15b3e5fd9cd0008d89d

See more details on using hashes here.

Provenance

The following attestation bundles were made for agy_headless_bridge-1.2.1-py3-none-any.whl:

Publisher: publish.yml on rhishi99/agy-headless-bridge

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page