EyeBrowse
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., "@EyeBrowseNavigate to https://news.ycombinator.com and summarize the top story"
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.
π¦ EyeBrowse
A stealthy, LLM-drivable browser engine β one codebase, two faces.
A Python library and an MCP server for driving a real, hard-to-detect browser β Camoufox, a Firefox fork with C++-level fingerprint spoofing β so legitimate automation isn't false-flagged or IP-banned by Cloudflare, DataDome, Akamai, or PerimeterX.

βΆ Higher-quality MP4: docs/demo.mp4 β real EyeBrowse captures, composed into an ad by examples/make_demo.py
Why EyeBrowse?
π₯· Stealth by default β engine-level fingerprint spoofing (
geoip+humanize+block_webrtcon out of the box);navigator.webdrivermasked; viewport auto-sized to the spoofed screen. Nopuppeteer-extraband-aids β the anti-detection is in the browser.π€ Built for LLMs β pages are read as an ARIA tree with
[ref=β¦]handles; the model acts by ref (click/type/hover), not by brittle CSS or raw pixels. Cross-origin iframes, shadow DOM, popups β handled.π§° Library and MCP from one codebase β a clean Python API (
EyeBrowse+Session), mirrored 1:1 by a thin MCP server (78browser_*tools) for Claude Code and any MCP client.πͺ Never boxed in β the curated high-level API doesn't hide Playwright: reach
session.page/.context/.browserfor anything it doesn't wrap.π Batteries included β multi-session, proxy + identity rotation, API-mode captcha solvers, full HAR capture, screen recording (MP4/GIF), and clean-markdown extraction.
Scope. EyeBrowse is a low-level browser engine β it holds no workflow logic. Consumers decide what to do; the engine provides what's possible.
Contents
Quickstart Β· Install Β· Features Β· Library Β· MCP Β· Proxy & identity Β· Extraction Β· Recording Β· How it works Β· Caveats Β· Tools Β· License
Quickstart
pip install eyebrowse
python -m camoufox fetch # one-time: download the stealth Firefox + GeoIP dbimport asyncio
from eyebrowse import EyeBrowse
async def main():
eb = EyeBrowse() # stealth defaults: geoip Β· humanize Β· block_webrtc
async with eb.session() as s:
await s.navigate("https://example.com")
print(await s.snapshot()) # ARIA tree with [ref=...] handles
await s.click("e6") # act on a ref from the snapshot
await eb.aclose()
asyncio.run(main())β¦or wire it into Claude Code (or any MCP client) β see Use over MCP.
Install
From PyPI
pip install eyebrowse # or: uv pip install eyebrowse
python -m camoufox fetch # one-time browser fetch (like Playwright's `playwright install`)
pip install "eyebrowse[extract]" # optional: + Crawl4AI markdown extraction (heavier)From source (development)
git clone https://github.com/Evil-Bane/eyebrowse && cd eyebrowse
uv sync # core engine (add --extra extract for Crawl4AI)
uv run python -m camoufox fetch
cp .env.example .env # only if you use a proxy / captcha keysPython 3.12 (pinned <3.13). Pinned engine: camoufox==0.4.11 (Firefox v135), playwright 1.60, mcp 1.27.
Features
π₯· Stealth | Camoufox engine-level fingerprint spoofing; |
π€ LLM interaction |
|
πͺ Frames & DOM | cross-origin iframe routing by ref, shadow-DOM piercing, popup/new-tab switching, |
π Multi-session | independent stealth sessions, each with its own context / identity / proxy. |
π Network | inspect requests/responses (incl. XHR/fetch bodies & WebSocket frames), block URLs, mock responses, go offline, full HAR export. |
πΎ State | cookies, localStorage & sessionStorage (CRUD), |
πͺͺ Identity rotation | fresh fingerprint (OS/screen) + isolated profile + paired proxy; pluggable residential |
π§© Captcha | pluggable API-mode solvers (CapSolver / 2Captcha / CapMonster / NextCaptcha) + TOTP β no browser extension. |
π Extraction | Crawl4AI |
π₯ Capture | screenshots, Playwright tracing, and screen recording β MP4 / WebM / GIF (ffmpeg). |
β Verify & debug | assertions, element highlighting, locator generation, geolocation/header emulation. |
Full per-tool reference: docs/TOOLS.md (78 tools across 17 groups).
Use as a library
import asyncio
from eyebrowse import EyeBrowse
async def main():
eb = EyeBrowse() # stealth defaults
try:
async with eb.session() as s: # a stealth session (auto-closed)
await s.navigate("https://example.com")
print(await s.snapshot()) # ARIA tree with [ref=...] handles
await s.click("e6") # act on a ref
await s.type("e8", "hello", submit=True)
png = await s.screenshot(full_page=True)
title = await s.page.title() # full Playwright power when you need it
finally:
await eb.aclose()
asyncio.run(main())Run the included proof: uv run python examples/direct_usage.py.
Use over MCP
EyeBrowse ships an MCP server (eyebrowse-mcp, FastMCP over stdio). Add it to any MCP client.
Claude Code (CLI):
claude mcp add eyebrowse -- eyebrowse-mcpAny MCP client (JSON config):
{
"mcpServers": {
"eyebrowse": {
"command": "eyebrowse-mcp"
}
}
}Then drive the loop: browser_navigate(url) β read the snapshot β act by ref
(browser_click / browser_type / β¦). A default session is auto-created, so most tools just
work. Full list: docs/TOOLS.md.
Proxy & identity (optional)
Runs proxyless by default (geoip still aligns locale/timezone to your real IP). Add a proxy
only when you want one:
await eb.new_session(proxy="http://user:pass@residential.example:8080")
await eb.rotate_identity(proxy="socks5://host:1080") # fresh fingerprint + paired IP
await eb.new_session(no_proxy=True) # force proxylessSet a default once via EYEBROWSE_PROXY_* in .env, eb.set_static_proxy(...), or a custom
ProxyProvider for rotation. Over MCP: browser_new_session(proxy_url=β¦) /
browser_new_identity(proxy_url=β¦) / browser_set_proxy(β¦).
Extraction
eb.extract() (or browser_extract) hands the rendered HTML to Crawl4AI's raw: feed and
returns clean, pruned markdown β no LLM is called and no LLM keys are ever read; the
consuming agent does any structuring.
md = await eb.extract() # markdown string
res = await eb.extract(output_path="data/page.md") # β {"path": ..., "chars": ...}Recording
Camoufox can't write native browser video, so EyeBrowse rolls its own recorder: it captures
viewport frames and encodes them with ffmpeg into a smooth, real-time MP4/WebM (and a
palette-optimized GIF that autoplays inline on GitHub). ffmpeg ships bundled
(imageio-ffmpeg), so it works out of the box.
await s.start_recording(fps=30)
# ... drive the browser ...
await s.stop_recording("demo.mp4", extra_paths=["demo.gif"])The demo at the top was produced this way β see examples/make_demo.py.
How it works
CONSUMERS ENGINE (library: eyebrowse/)
Claude Code ββMCPβββΆ mcp/ βββΆ EyeBrowse faΓ§ade (public API)
your code β import βββββββββββΆ ββ Camoufox engine (stealth context, HAR)
any MCP client ββ proxy / identity rotation (pluggable)
ββ captcha solvers (pluggable, API-mode)
ββ Crawl4AI (raw: feed) β clean markdownThe faΓ§ade (EyeBrowse + Session) is the product; the MCP adapter is a thin 1:1 wrapper over
it. The high-level API is curated and LLM-friendly β not a reimplementation of all of Playwright
β and the raw page / context / browser objects are always one attribute away.
Camoufox caveats
These shape the API β worth knowing:
evaluateruns in an isolated world β it sees the DOM but not page-scriptwindow.*globals. Read app state via DOM / network / HAR.reload / back / forward aren't reported by Firefox's Juggler, so they're driven via
goto+ an internal navigate() history (they traverse navigate()-driven history, not click-driven).Not available on Firefox/Camoufox (intentionally omitted, not broken):
page.pdf(), media emulation, and native Playwright video β use the ffmpeg recorder instead.
Project layout
eyebrowse/
api.py EyeBrowse faΓ§ade β the single public entry point
config.py settings / secrets (pydantic-settings)
snapshot.py aria_snapshot(mode="ai") + aria-ref= resolution
proxy.py ProxyConfig + pluggable ProxyProvider
identity.py Identity + random_identity()
extract.py Crawl4AI raw: feed β markdown (lazy, optional dep)
engine/ camoufox_engine.py (launch) + session.py (verbs + registry)
captcha/ solver ABC + 4 providers + DOM detect/inject
mcp/ FastMCP server + state + tools/ (17 groups, 78 tools)
examples/direct_usage.py library proof (no MCP)
examples/make_demo.py the screen-recording demo above
docs/TOOLS.md full tool referenceBuild notes, version-pin rationale, and verified Camoufox behavior live in CLAUDE.md.
Use responsibly
EyeBrowse drives a real browser with anti-detection features. Use it only against sites you own or are explicitly authorized to automate, and within their terms and applicable law.
License
MIT Β© Evil-Bane
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
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/Evil-Bane/eyebrowse'
If you have feedback or need assistance with the MCP directory API, please join our Discord server