blop lets you test any web application by just describing what you want to test in plain English. It opens a real browser, clicks through your app like a human would, and tells you what's broken — complete with screenshots, severity labels, and suggested fixes.
You don't write test code. You just talk to it.
- Discovers your app's key flows by crawling it (login, pricing, contact, checkout, etc.)
- Records those flows by watching an AI agent use your app
- Replays them on demand to catch regressions
- Reports exactly what broke, how severe it is, and what to fix
It plugs into Cursor or Claude Code as an MCP tool — meaning you just ask it to run tests in a chat window, the same way you'd ask a colleague.
| What | Where to get it | Takes |
|---|---|---|
| Python 3.11 or newer | python.org/downloads | 5 min |
uv (fast Python installer) |
Run curl -LsSf https://astral.sh/uv/install.sh | sh in Terminal |
1 min |
| Google API key (free tier works) | aistudio.google.com/app/apikey | 2 min |
| Cursor or Claude Code | cursor.com or npm i -g @anthropic-ai/claude-code |
5 min |
Open Terminal and run these commands one at a time:
# 1. Go into the blop folder
cd /path/to/blop-use
# 2. Create a Python environment
uv venv && source .venv/bin/activate
# 3. Install blop
uv pip install -e .
# 4. Install the browser blop controls
playwright install chromium --with-deps --no-shellYou should see no errors. If you do, check the Troubleshooting section below.
Copy the example config file and fill it in:
cp .env.example .envThen open .env in any text editor and fill in your values:
# Required — get this from aistudio.google.com/app/apikey
GOOGLE_API_KEY=your_key_here
# Your app's URL
APP_BASE_URL=https://your-app.com
# Login details (only needed for testing authenticated pages)
LOGIN_URL=https://your-app.com/login
TEST_USERNAME=your@email.com
TEST_PASSWORD=your_password
Everything else can stay as-is.
- Open Cursor
- Go to Settings → MCP
- Click Add MCP Server and paste this (update the path to where you cloned blop):
{
"mcpServers": {
"blop": {
"command": "uv",
"args": ["--directory", "/path/to/blop-use", "run", "python", "-m", "blop.server"],
"env": {
"GOOGLE_API_KEY": "your_key_here",
"APP_BASE_URL": "https://your-app.com",
"LOGIN_URL": "https://your-app.com/login",
"TEST_USERNAME": "your@email.com",
"TEST_PASSWORD": "your_password"
}
}
}
}- Restart Cursor. You should see blop listed as a connected tool in Settings → MCP.
Run this once in your terminal:
claude mcp add blop /path/to/blop-use/.venv/bin/blop-mcp \
-e GOOGLE_API_KEY="your_key_here" \
-e APP_BASE_URL="https://your-app.com" \
-e LOGIN_URL="https://your-app.com/login" \
-e TEST_USERNAME="your@email.com" \
-e TEST_PASSWORD="your_password"Type /mcp in Claude Code to verify — you should see blop: connected.
Open a chat window in Cursor or Claude Code and paste this (swap in your app URL):
Use blop to test https://your-app.com
1. Optionally call validate_setup(app_url="https://your-app.com") to confirm the environment is ready
2. Call explore_site_inventory to map the interface first (routes/forms/buttons + compact page structure)
3. Call discover_test_flows to find what flows exist on the site
4. Run a regression test on all of them
5. Poll get_test_results until status is "completed", then show me the results with severity levels
That's it. blop will crawl the site, record the flows, run the tests, and report back — all without you writing a single line of test code.
Crawls your app and asks AI to figure out what the important user journeys are. Returns a list of suggested flows with descriptions of what to verify.
Basic usage:
discover_test_flows("https://your-app.com")
With more context (gets better results):
discover_test_flows(
app_url="https://your-app.com",
business_goal="Find all revenue-critical flows like checkout and upgrade",
max_depth=2
)
Parameters:
| Parameter | What it does | Example |
|---|---|---|
app_url |
The website to scan | "https://app.example.com" |
business_goal |
Tell it what matters most to your business | "Focus on checkout and onboarding" |
profile_name |
Use a logged-in account to scan private pages | "my-auth-profile" |
max_depth |
How deep to crawl (1 = homepage only, 2 = homepage + linked pages) | 2 |
max_pages |
Max pages to crawl before planning flows | 20 |
seed_urls |
Start crawl from specific same-origin URLs | ["https://app.example.com/pricing"] |
include_url_pattern |
Regex: only crawl URLs that match | `"/(pricing |
exclude_url_pattern |
Regex: skip noisy URLs | `"/(blog |
return_inventory |
Include raw crawl inventory in response | true |
command |
Free-text instruction — blop figures out the rest | "Discover auth flows for the dashboard" |
What you get back: Flows include a business_criticality hint (revenue, activation, retention, support, other) so you can prioritize recording and triage results.
{
"flows": [
{
"flow_name": "user_login",
"goal": "User logs in with email and password and reaches the dashboard",
"severity_if_broken": "blocker",
"confidence": 0.92,
"business_criticality": "activation"
},
{
"flow_name": "pricing_page_upgrade",
"goal": "Visitor views pricing, clicks Pro plan CTA, reaches checkout",
"severity_if_broken": "high",
"confidence": 0.85,
"business_criticality": "revenue"
}
],
"inventory_summary": {
"auth_signals": ["sign in", "/login"],
"business_signals": ["pricing", "checkout"]
},
"quality": {
"passed": true,
"warnings": []
}
}Runs inventory-only discovery (no Gemini flow planning) so you can inspect routes, forms, headings, auth signals, and business signals first. It now also includes page_structures, a compact per-page list of interactive ARIA nodes (role + name) to give agents layout context.
explore_site_inventory(
app_url="https://your-app.com",
max_depth=2,
max_pages=20,
include_url_pattern="/(pricing|signup|dashboard)"
)
Use this when you want deterministic topology mapping before discover_test_flows.
Captures a single-page interactive structure snapshot using Playwright's accessibility tree. Useful before recording or debugging when you want context for one URL without running a full crawl.
get_page_structure(
app_url="https://your-app.com",
url="https://your-app.com/pricing", # optional; defaults to app_url
profile_name="my-auth-profile" # optional
)
Returns a flattened interactive_nodes list so MCP agents can reason about what controls are available before choosing actions.
Saves your login details so blop can test pages that require being signed in. Your password is only stored locally on your machine.
Basic usage (username + password from your .env file):
save_auth_profile(
profile_name="my-app-login",
auth_type="env_login",
login_url="https://your-app.com/login"
)
Auth types explained:
| Type | When to use | Example |
|---|---|---|
env_login |
You have a username + password | Standard email/password login |
storage_state |
You have a browser session file from Playwright | SSO, OAuth, MFA flows |
cookie_json |
You have exported browser cookies | When you can't automate login |
Tips:
- blop caches sessions for 1 hour — it won't re-login every time you run tests
- Your credentials are read from environment variables, never stored as plain text in the database
- For SSO/Google login, use
storage_stateor the interactivecapture_auth_sessiontool (see below) - Use
user_data_dirwhen the login provider (e.g. Google, LinkedIn) blocks fresh browser contexts — blop will use a persistent Chromium profile
Opens a visible browser at your login URL. You complete Google/GitHub OAuth, MFA, or any flow by hand. The tool polls the page URL every 500ms and, when it detects success, saves the Playwright storage state and creates an auth profile automatically. No manual session export needed.
Basic usage:
capture_auth_session(
profile_name="my-app-sso",
login_url="https://your-app.com/login",
success_url_pattern="/dashboard"
)
Parameters:
| Parameter | What it does | Example |
|---|---|---|
profile_name |
Name for the saved auth profile | "my-app-sso" |
login_url |
URL to open (your app's login or OAuth start) | "https://app.example.com/login" |
success_url_pattern |
URL substring that means "logged in" (optional) | "/dashboard" — if omitted, any URL change away from the login page counts as success |
timeout_secs |
Max seconds to wait for you to complete login (default 120) | 180 |
user_data_dir |
Path to a persistent Chromium profile dir (optional) | .blop/chrome_profile_myapp — use when OAuth providers treat a fresh browser as a bot |
Returns: status is "captured" (session saved, profile ready for record_test_flow and run_regression_test) or "timeout" (no success detected in time). On success you get storage_state_path; the profile is already stored — just pass profile_name to other tools.
Runs an AI agent in a real browser to accomplish a goal, and saves every step it takes. You can then replay this recording as many times as you want.
Basic usage:
record_test_flow(
app_url="https://your-app.com",
flow_name="user_signup",
goal="Sign up for a new account with email and verify the welcome screen appears"
)
With authentication and business criticality (for flows behind login):
record_test_flow(
app_url="https://your-app.com",
flow_name="create_new_project",
goal="Log in, create a new project called 'Test Project', and verify it appears in the project list",
profile_name="my-app-login",
business_criticality="revenue"
)
business_criticality (optional) is one of revenue, activation, retention, support, other. It is used in results and severity labels (e.g. "BLOCKER in revenue flow: checkout") so you can triage by business impact.
Writing good goals — what makes the difference:
| ❌ Vague (gets generic results) | ✅ Specific (gets reliable tests) |
|---|---|
"Test the login" |
"Log in with email and password, verify the dashboard shows the user's name in the top right corner" |
"Check pricing" |
"Navigate to pricing, verify Free, Pro ($35/mo), and Enterprise tiers are visible, click the Pro CTA and confirm it leads to checkout" |
"Test the form" |
"Fill in the contact form with name, company email, and message, submit it, and verify a confirmation message appears" |
What gets captured per step:
- The element clicked or filled (selector + visible text)
- A screenshot at that moment
- The URL before and after
- Final assertions generated by AI from the end state
Replays recorded flows against your app. Returns immediately with a run_id — poll for results with get_test_results. Run status moves: queued → running → completed or failed. If the auth profile cannot be resolved, status is waiting_auth and no flows run until you fix the profile and retry.
Basic usage:
run_regression_test(
app_url="https://your-app.com",
flow_ids=["abc123", "def456"]
)
With auth and hybrid mode:
run_regression_test(
app_url="https://your-app.com",
flow_ids=["abc123", "def456"],
profile_name="my-app-login",
run_mode="hybrid"
)
Run modes explained:
| Mode | What it does | When to use |
|---|---|---|
hybrid (default) |
Tries saved steps first; if a selector breaks, AI repairs that single step | Best for most cases |
strict_steps |
Follows saved steps exactly — fails immediately if anything doesn't match | CI/CD where you want strict enforcement |
goal_fallback |
Ignores saved steps, replays using only the original goal description | When the UI has changed significantly |
Polls a running test and returns structured results. Call it repeatedly until status is "completed", "failed", or "waiting_auth". When status is waiting_auth, the response includes waiting_auth_message explaining that the auth profile could not be resolved (check credentials or re-run capture_auth_session if the session expired).
get_test_results("your-run-id-here")
Understanding the results:
{
"status": "completed",
"severity_counts": {
"blocker": 1,
"high": 0,
"medium": 2,
"pass": 4
},
"failed_cases": [
{
"flow_name": "checkout_flow",
"severity": "blocker",
"replay_mode": "hybrid_repair",
"step_failure_index": 3,
"assertion_failures": ["Payment page should load after clicking Subscribe"],
"repro_steps": ["Go to /pricing", "Click 'Get Pro Plan'", "Observe: redirected to homepage instead of checkout"]
}
]
}Severity levels — what they mean for your business:
| Level | Meaning | Action |
|---|---|---|
| 🔴 blocker | Core workflow is completely broken. Users can't complete a key task. | Fix before shipping anything |
| 🟠 high | Major feature broken, significant user impact | Fix in current sprint |
| 🟡 medium | Partial issue, workaround possible | Fix in next sprint |
| 🟢 low | Cosmetic or edge case | Backlog |
| ✅ pass | Everything worked | No action needed |
replay_mode tells you how the test ran:
strict_steps— selector matched, step ran exactly as recordedhybrid_repair— original selector broke, AI found the element another waygoal_fallback— step-by-step replay failed entirely, fell back to full agent replay
Lists recent regression runs, optionally filtered by status.
list_runs(limit=20, status="running")
Useful when you lost a run_id or need to inspect background runs.
Returns control-plane run events (queued, started, per-case completion, completed/failed) so you can inspect lifecycle and timing without opening artifacts first.
get_run_health_stream(
run_id="your-run-id",
limit=500 # optional
)
Useful for quick triage when a run exits unexpectedly or when you want to inspect replay/healing metadata at event granularity.
Aggregates recent runs into high-signal diagnostics:
- flaky step leaderboard
- top failing transitions
- failure rates by
business_criticality(revenue/activation/retention/support/other)
get_risk_analytics(limit_runs=30)
Use this to prioritize stabilization work across many runs instead of triaging one run at a time.
Lists every flow you've ever recorded.
list_recorded_tests()
Returns a list with flow_id, flow_name, app_url, goal, and created_at. Use the flow_id values in run_regression_test.
Re-runs a specific failed case in a visible browser window (not headless), captures every screenshot, and generates a plain-English explanation of what went wrong.
debug_test_case(
run_id="your-run-id",
case_id="the-failed-case-id"
)
What you get back:
{
"status": "fail",
"step_failure_index": 3,
"replay_mode": "hybrid_repair",
"assertion_failures": ["Dashboard should show user inbox after login"],
"why_failed": "The login succeeded but the session cookie was not persisted between the auth step and the dashboard navigation. The app redirected to /login again instead of /dashboard.",
"repro_steps": ["Navigate to /auth", "Fill email and password", "Click Sign In", "Session lost — redirect back to /auth"],
"screenshots": ["runs/screenshots/run123/case456/step_003.png"]
}Checks preconditions before you run flows: GOOGLE_API_KEY, Chromium installed, SQLite DB, optional app_url reachability, and optional auth profile (including whether a storage_state session is still valid). Use it after changing env vars or before a big run.
validate_setup(app_url="https://your-app.com", profile_name="my-app-login")
Returns: status is "ready" (all checks passed), "warnings" (e.g. app URL unreachable but you can still run), or "blocked" (e.g. missing API key or Chromium). The checks array lists each condition and whether it passed; blockers and warnings give short messages. If an auth profile's session has expired, the message will suggest re-running capture_auth_session.
blop now exposes read-only MCP resources so agents can pull structured context without triggering heavy tool workflows.
Latest saved inventory for an app URL.
- URL-encode the full app URL in
{app}. - Example:
blop://inventory/https%3A%2F%2Fapp.example.com
Latest persisted SiteContextGraph snapshot (nodes, edges, archetype, freshness/confidence metadata).
- Example:
blop://context-graph/https%3A%2F%2Fapp.example.com
Artifact index for a run (artifact metadata + case ids), useful before drilling into screenshots/traces.
- Example:
blop://run/abc123/artifact-index
Flow-level stability profile derived from historical cases (pass/failure rates, replay-mode distribution, stability score).
- Example:
blop://flow/def456/stability-profile
- Read
inventory+context-graphresources. - Run
discover_test_flows/record_test_flowusing that context. - Run
run_regression_test. - Read
artifact-index+stability-profile. - Use
get_risk_analyticsfor cross-run prioritization.
blop v2 expands beyond regression execution into change intelligence, journey health, incident clustering, and remediation orchestration.
blop_v2_get_surface_contract— returns machine-readable request/response schemas + examples for all v2 tools.blop_v2_capture_context— captures a context graph snapshot and structural diff summary.blop_v2_compare_context— compares two graph versions and returns structural/business impact.blop_v2_assess_release_risk— release-level risk score and top risks from context/run evidence.blop_v2_get_journey_health— SLO-like health view for business journeys over time.blop_v2_cluster_incidents— deduplicates failures into incident clusters with blast radius.blop_v2_generate_remediation— emits issue-ready remediation drafts (repro + evidence).blop_v2_ingest_telemetry_signals— ingests external signals (error rate/latency/conversion).blop_v2_get_correlation_report— correlates failures with telemetry changes for prioritization.
blop://v2/contracts/toolsblop://v2/context/{urlencoded_app_url}/latestblop://v2/context/{urlencoded_app_url}/history/{limit}blop://v2/context/{urlencoded_app_url}/diff/{baseline_graph_id}/{candidate_graph_id}blop://v2/release/{release_id}/risk-summaryblop://v2/journey/{urlencoded_app_url}/health/{window}(window:24h,7d,30d)blop://v2/incidents/{urlencoded_app_url}/openblop://v2/incident/{cluster_id}blop://v2/incident/{cluster_id}/remediation-draftblop://v2/correlation/{urlencoded_app_url}/{window}
- v1 tools remain supported (
discover_test_flows,run_regression_test,get_test_results, etc.). - v1 responses now include
related_v2_resourceslinks so agents can progressively adopt v2 context. - v2 resources use a stable envelope:
{
"resource_version": "v2",
"generated_at": "2026-03-18T12:00:00Z",
"app_url": "https://app.example.com",
"data": {}
}Use blop to discover and test https://new-saas-app.com from scratch.
1. Call discover_test_flows with business_goal="Find all revenue-critical flows including signup, checkout, and onboarding"
2. Record the top 5 flows it finds
3. Run regression on all of them
4. Show me anything with severity "high" or "blocker"
We're about to ship a new version. Use blop to run a pre-release check on https://staging.myapp.com:
1. list_recorded_tests — show what flows we have
2. run_regression_test on all flows against staging
3. get_test_results — compare pass/fail to last run
4. debug_test_case on anything that changed from pass to fail
Test the authenticated product experience on https://app.myapp.com:
1. save_auth_profile("prod-user", "env_login", login_url="https://app.myapp.com/login")
2. record_test_flow for: dashboard load, core feature (e.g. "create new project"), settings page, and billing/upgrade flow
3. run_regression_test with profile_name="prod-user"
4. get_test_results — show me the full breakdown
A user reported the checkout button isn't working. Use blop to investigate:
1. record_test_flow("https://myapp.com", "checkout_bug", "Navigate to pricing, click the Pro plan CTA, and verify checkout loads")
2. run_regression_test on that flow
3. get_test_results — check step_failure_index and assertion_failures
4. debug_test_case on the failure to get screenshots and a plain-English explanation
Option A — Interactive capture (recommended): Use capture_auth_session so blop opens a browser and you log in once; it saves the session and creates the profile automatically.
capture_auth_session(
profile_name="sso-session",
login_url="https://your-app.com/login",
success_url_pattern="/dashboard",
timeout_secs=120
)
For providers (e.g. Google, LinkedIn) that block "headless" or fresh contexts, use a persistent profile:
capture_auth_session(
profile_name="sso-session",
login_url="https://your-app.com/login",
success_url_pattern="/dashboard",
user_data_dir=".blop/chrome_profile_myapp"
)
Option B — Manual export: Log in manually, export the session (e.g. Playwright or Chrome DevTools → Application → Storage), then:
save_auth_profile(
profile_name="sso-session",
auth_type="storage_state",
storage_state_path="/path/to/my-session.json"
)
Your .env file blop
─────────────────────────────────────────────────────────────
TEST_USERNAME=user@company.com → Reads at runtime
TEST_PASSWORD=secret → Never stored in DB
↓
Opens login page
Fills credentials
Saves session cookie
↓
.blop/auth_state_profile.json
(valid for 1 hour, then re-logs in)
blop tries these login field selectors automatically, in order:
input[name="username"]input[name="email"]input[type="email"]#email- Any input with "email" in the placeholder
You can override with env vars TEST_USERNAME_SELECTOR and TEST_PASSWORD_SELECTOR if your login form is unusual.
your-project/
├── .env ← Your credentials (never commit this)
├── .blop/
│ ├── runs.db ← All test history (SQLite)
│ └── auth_state_*.json ← Cached login sessions
└── runs/
├── screenshots/
│ └── <run_id>/<case_id>/
│ ├── step_000.png ← Screenshot at each step
│ └── step_001.png
├── traces/
│ └── <run_id>/<case_id>.zip ← Playwright trace (open with trace viewer)
└── console/
└── <run_id>/<case_id>.log ← Browser console errors
"blop not found" or "command not found"
Make sure you've activated the virtual environment: source .venv/bin/activate
"GOOGLE_API_KEY not set"
Check your .env file exists in the blop-use folder and has your key on the GOOGLE_API_KEY= line. Alternatively, set it directly in the MCP config JSON.
Login keeps failing
- Double-check your
TEST_USERNAMEandTEST_PASSWORDin.env - Try visiting your
LOGIN_URLmanually to confirm the credentials work - If your login uses unusual field names, add
TEST_USERNAME_SELECTOR=input[name="your-field"]to.env - For SSO/MFA, use
capture_auth_session(opens a browser so you log in once; session is saved automatically) orauth_type="storage_state"with an exported session file - Run
validate_setup(profile_name="your-profile")to verify the profile; if the session expired, re-runcapture_auth_sessionor refresh your storage state file
Tests are all passing but you know something is broken
The regression engine uses AI vision to evaluate assertions — it shouldn't produce false positives. If something is marked pass that looks wrong, use debug_test_case to re-run it with full screenshot capture and see what the browser actually showed.
"MCP server not connected" in Cursor
- Check the path in
mcp.jsonpoints to where you actually cloned blop-use - Make sure
uvis installed (which uvin Terminal) - Restart Cursor fully (Cmd+Q, reopen)
The browser opens but does nothing / hangs
Your BLOP_MAX_STEPS limit (default 50) may be too low for complex flows. Add BLOP_MAX_STEPS=100 to .env.
blop uses a deterministic-first architecture with adaptive repair fallback. For easier tuning across different interfaces, you can choose a profile:
BLOP_EXPLORATION_PROFILE=default— balanced defaults for most apps.BLOP_EXPLORATION_PROFILE=saas_marketing— tuned for async SPAs, heavy client-side editors, and cross-origin handoffs likerendley.com→app.rendley.com.
You can still override individual knobs with env vars (BLOP_NETWORK_IDLE_WAIT, BLOP_SPA_SETTLE_MS, BLOP_AGENT_MAX_FAILURES, BLOP_AGENT_MAX_ACTIONS_PER_STEP, BLOP_DISCOVERY_MAX_PAGES).
Design baseline references:
- Playwright deterministic web-first model: Playwright intro, playwright-mcp
- Agentic fallback model: browser-use
- Healing + observability lifecycle: TestSprite create tests, test lifecycle, healing, dashboard
- MCP server/tool contract design: server concepts, build server
Use this quick matrix to validate that the architecture stays generic while working well for Rendley-like products:
validate_setup(app_url="https://rendley.com")explore_site_inventory(app_url="https://rendley.com", max_pages=20, include_url_pattern="/(pricing|tools|start|create)")discover_test_flows(app_url="https://rendley.com", max_pages=20, return_inventory=true, business_goal="prioritize onboarding, video creation, and pricing conversion")- Record 3-5 flow families (landing conversion, onboarding/auth handoff, project/workspace actions, pricing/plan gating).
- Run one
hybridreplay, one strict replay (strict_steps), and one fallback replay (goal_fallback) to verify graceful degradation. - Use
list_runs(status="running")+get_test_results(run_id=...)and verify failure class + next actions are actionable.
| Variable | Required | Default | What it does |
|---|---|---|---|
GOOGLE_API_KEY |
Yes | — | Gemini API key. Get one free at aistudio.google.com |
APP_BASE_URL |
No | — | Default app URL (used as fallback if no URL passed to tools) |
LOGIN_URL |
No | — | Where blop navigates to log in |
TEST_USERNAME |
No | — | Login email/username |
TEST_PASSWORD |
No | — | Login password |
TEST_USERNAME_SELECTOR |
No | auto-detected | CSS selector for the username input field |
TEST_PASSWORD_SELECTOR |
No | auto-detected | CSS selector for the password input field |
STORAGE_STATE_PATH |
No | — | Path to a saved Playwright session (for SSO/OAuth) |
COOKIE_JSON_PATH |
No | — | Path to exported browser cookies (JSON array) |
BLOP_DB_PATH |
No | .blop/runs.db |
Where blop stores its database |
BLOP_HEADLESS |
No | true |
false = show browser window during tests (useful for debugging) |
BLOP_MAX_STEPS |
No | 50 |
Max steps the AI agent takes per flow |
BLOP_EXPLORATION_PROFILE |
No | default |
Tuning preset (default or saas_marketing) for discovery and replay behavior |
BLOP_DISCOVERY_MAX_PAGES |
No | profile-driven | Default crawl page cap for discovery tools |
BLOP_AGENT_MAX_FAILURES |
No | profile-driven | Max recoverable action failures before agent aborts recording |
BLOP_AGENT_MAX_ACTIONS_PER_STEP |
No | profile-driven | Max agent actions per reasoning step during recording |
BLOP_NETWORK_IDLE_WAIT |
No | 2.0 |
Seconds to wait for network idle after page load (increase for WebGL/WASM or slow dashboards) |
BLOP_SPA_SETTLE_MS |
No | 1500 |
Extra settle time in ms after SPA navigation (for pushState / client-side routing) |
- Python 3.11+
- Google API key with Gemini access (free tier works)
- Cursor or Claude Code with MCP support
- macOS, Linux, or Windows with WSL2
Powered by Browser Use and Google Gemini