agy-headless-bridge MCP server
Allows interaction with Google's Antigravity CLI (agy) through a pseudo-terminal, enabling AI agents to ask prompts and perform research using Gemini models via the Antigravity platform in headless environments.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@agy-headless-bridge MCP serverask: explain closures in one line"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
agy-headless-bridge
Call the Google Antigravity CLI (agy) headlessly โ and actually get output back.
Codename PtyGravity ยท pty + antiGravity
๐ 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 "prompt""]
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 "prompt"<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 |
| embedding agy in Python |
CLI |
| shell scripts, quick calls |
MCP server |
| letting an agent call agy as a tool |
Related MCP server: agy-mcp
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"| APlatform | pty backend | Status |
Windows | ConPTY via | โ verified (agy 1.0.6) |
Linux / macOS | stdlib | โ ๏ธ pty mechanics verified on Linux CI; real |
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
agyClaude Code plugins? They wrapagyfor triggering (slash commands, model selection) but still callagy -pdirectly โ 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:
Python 3.9+ โ
python --version.The Antigravity CLI (
agy), installed and authenticated:Install: https://antigravity.google/cli
Authenticate once interactively (
agyopens a browser OAuth flow), or setANTIGRAVITY_API_KEYin 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.)
Windows only:
pywinpty(installed automatically as a dependency). POSIX uses the stdlibptymodule โ nothing extra.
This package does not install or authenticate
agy, and does not bundle any credentials. It only spawns theagyalready on your machine.
Install
Requires Python 3.9+.
pip install agy-headless-bridge # pywinpty auto-installs on Windows onlyFrom 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 |
| cwd (auto) | Add a dir to agy's workspace (repeatable). |
| off | Don't auto-add cwd (use for research / Q&A). |
| agy default | agy |
|
| Hard ceiling (absolute wall). |
|
| Kill after this many seconds of no output. |
MCP server
claude mcp add --transport stdio antigravity -- agy-mcp-serveragy-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.
claude mcp add --transport stdio antigravity -- \
python -m agy_headless_bridge.mcp_serverWindows: use
py -3.11(the Python Launcher) instead ofpythonin the command above. Barepythoncan resolve to the wrong interpreter or the Windows Store stub, which surfaces as an-32000MCP 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 |
|
| string | โ | one-shot prompt sent to agy |
|
|
|
| |
|
| string[] | explicit dirs for agy's workspace (overrides the | |
|
| string | agy | |
|
| number | hard timeout override, seconds | |
|
| 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 โ |
A CI step runs an |
|
A Python pipeline fans work out to agy |
|
Codex / any MCP-capable agent delegates to agy | register the same MCP server |
Cron / scheduled job summarizes logs via agy |
|
Wire into Claude Code
Register the MCP server, then prompt Claude to use it:
claude mcp add --transport stdio antigravity -- agy-mcp-serverPrompt to Claude Code: "Use the
agy_asktool 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_researchwith 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 |
| auto-detect | Absolute path to the |
|
| Hard ceiling (absolute wall), in seconds |
|
| 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[0mTroubleshooting / 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]"
pytestUnit 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
agysettings.jsonconcern, covered by theantigravity-ccplugin. 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
agywithin 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.
Maintenance
Tools
Latest Blog Posts
- Your AI Chatbot Just Exposed Your CEO's Salary to an InternBy Om-Shree-0709 on .Agent IdentityMCP SecurityOAuth Delegation
- Why MCP Servers Need Execution Sandboxing (And Why Your Current Stack Isn't Enough)By Om-Shree-0709 on .Agentic AiPrompt InjectionWebAssembly
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/rhishi99/agy-headless-bridge'
If you have feedback or need assistance with the MCP directory API, please join our Discord server