* feat: switch managed browser provider from Browserbase to Browser Use
The Nous subscription tool gateway now routes browser automation through
Browser Use instead of Browserbase. This commit:
- Adds managed Nous gateway support to BrowserUseProvider (idempotency
keys, X-BB-API-Key auth header, external_call_id persistence)
- Removes managed gateway support from BrowserbaseProvider (now
direct-only via BROWSERBASE_API_KEY/BROWSERBASE_PROJECT_ID)
- Updates browser_tool.py fallback: prefers Browser Use over Browserbase
- Updates nous_subscription.py: gateway vendor 'browser-use', auto-config
sets cloud_provider='browser-use' for new subscribers
- Updates tools_config.py: Nous Subscription entry now uses Browser Use
- Updates setup.py, cli.py, status.py, prompt_builder.py display strings
- Updates all affected tests to match new behavior
Browserbase remains fully functional for users with direct API credentials.
The change only affects the managed/subscription path.
* chore: remove redundant Browser Use hint from system prompt
* fix: upgrade Browser Use provider to v3 API
- Base URL: api/v2 -> api/v3 (v2 is legacy)
- Unified all endpoints to use native Browser Use paths:
- POST /browsers (create session, returns cdpUrl)
- PATCH /browsers/{id} with {action: stop} (close session)
- Removed managed-mode branching that used Browserbase-style
/v1/sessions paths — v3 gateway now supports /browsers directly
- Removed unused managed_mode variable in close_session
* fix(browser-use): use X-Browser-Use-API-Key header for managed mode
The managed gateway expects X-Browser-Use-API-Key, not X-BB-API-Key
(which is a Browserbase-specific header). Using the wrong header caused
a 401 AUTH_ERROR on every managed-mode browser session create.
Simplified _headers() to always use X-Browser-Use-API-Key regardless
of direct vs managed mode.
* fix(nous_subscription): browserbase explicit provider is direct-only
Since managed Nous gateway now routes through Browser Use, the
browserbase explicit provider path should not check managed_browser_available
(which resolves against the browser-use gateway). Simplified to direct-only
with managed=False.
* fix(browser-use): port missing improvements from PR #5605
- CDP URL normalization: resolve HTTP discovery URLs to websocket after
cloud provider create_session() (prevents agent-browser failures)
- Managed session payload: send timeout=5 and proxyCountryCode=us for
gateway-backed sessions (prevents billing overruns)
- Update prompt builder, browser_close schema, and module docstring to
replace remaining Browserbase references with Browser Use
- Dynamic /browser status detection via _get_cloud_provider() instead
of hardcoded env var checks (future-proof for new providers)
- Rename post_setup key from 'browserbase' to 'agent_browser'
- Update setup hint to mention Browser Use alongside Browserbase
- Add tests: CDP normalization, browserbase direct-only guard,
managed browser-use gateway, direct browserbase fallback
---------
Co-authored-by: rob-maron <132852777+rob-maron@users.noreply.github.com>
218 lines
7.8 KiB
Python
218 lines
7.8 KiB
Python
"""Browserbase cloud browser provider (direct credentials only)."""
|
|
|
|
import logging
|
|
import os
|
|
import uuid
|
|
from typing import Any, Dict, Optional
|
|
|
|
import requests
|
|
|
|
from tools.browser_providers.base import CloudBrowserProvider
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class BrowserbaseProvider(CloudBrowserProvider):
|
|
"""Browserbase (https://browserbase.com) cloud browser backend.
|
|
|
|
This provider requires direct BROWSERBASE_API_KEY and BROWSERBASE_PROJECT_ID
|
|
credentials. Managed Nous gateway support has been removed — the Nous
|
|
subscription now routes through Browser Use instead.
|
|
"""
|
|
|
|
def provider_name(self) -> str:
|
|
return "Browserbase"
|
|
|
|
def is_configured(self) -> bool:
|
|
return self._get_config_or_none() is not None
|
|
|
|
# ------------------------------------------------------------------
|
|
# Session lifecycle
|
|
# ------------------------------------------------------------------
|
|
|
|
def _get_config_or_none(self) -> Optional[Dict[str, Any]]:
|
|
api_key = os.environ.get("BROWSERBASE_API_KEY")
|
|
project_id = os.environ.get("BROWSERBASE_PROJECT_ID")
|
|
if api_key and project_id:
|
|
return {
|
|
"api_key": api_key,
|
|
"project_id": project_id,
|
|
"base_url": os.environ.get("BROWSERBASE_BASE_URL", "https://api.browserbase.com").rstrip("/"),
|
|
}
|
|
return None
|
|
|
|
def _get_config(self) -> Dict[str, Any]:
|
|
config = self._get_config_or_none()
|
|
if config is None:
|
|
raise ValueError(
|
|
"Browserbase requires BROWSERBASE_API_KEY and BROWSERBASE_PROJECT_ID "
|
|
"environment variables."
|
|
)
|
|
return config
|
|
|
|
def create_session(self, task_id: str) -> Dict[str, object]:
|
|
config = self._get_config()
|
|
|
|
# Optional env-var knobs
|
|
enable_proxies = os.environ.get("BROWSERBASE_PROXIES", "true").lower() != "false"
|
|
enable_advanced_stealth = os.environ.get("BROWSERBASE_ADVANCED_STEALTH", "false").lower() == "true"
|
|
enable_keep_alive = os.environ.get("BROWSERBASE_KEEP_ALIVE", "true").lower() != "false"
|
|
custom_timeout_ms = os.environ.get("BROWSERBASE_SESSION_TIMEOUT")
|
|
|
|
features_enabled = {
|
|
"basic_stealth": True,
|
|
"proxies": False,
|
|
"advanced_stealth": False,
|
|
"keep_alive": False,
|
|
"custom_timeout": False,
|
|
}
|
|
|
|
session_config: Dict[str, object] = {"projectId": config["project_id"]}
|
|
|
|
if enable_keep_alive:
|
|
session_config["keepAlive"] = True
|
|
|
|
if custom_timeout_ms:
|
|
try:
|
|
timeout_val = int(custom_timeout_ms)
|
|
if timeout_val > 0:
|
|
session_config["timeout"] = timeout_val
|
|
except ValueError:
|
|
logger.warning("Invalid BROWSERBASE_SESSION_TIMEOUT value: %s", custom_timeout_ms)
|
|
|
|
if enable_proxies:
|
|
session_config["proxies"] = True
|
|
|
|
if enable_advanced_stealth:
|
|
session_config["browserSettings"] = {"advancedStealth": True}
|
|
|
|
# --- Create session via API ---
|
|
headers = {
|
|
"Content-Type": "application/json",
|
|
"X-BB-API-Key": config["api_key"],
|
|
}
|
|
|
|
response = requests.post(
|
|
f"{config['base_url']}/v1/sessions",
|
|
headers=headers,
|
|
json=session_config,
|
|
timeout=30,
|
|
)
|
|
|
|
proxies_fallback = False
|
|
keepalive_fallback = False
|
|
|
|
# Handle 402 — paid features unavailable
|
|
if response.status_code == 402:
|
|
if enable_keep_alive:
|
|
keepalive_fallback = True
|
|
logger.warning(
|
|
"keepAlive may require paid plan (402), retrying without it. "
|
|
"Sessions may timeout during long operations."
|
|
)
|
|
session_config.pop("keepAlive", None)
|
|
response = requests.post(
|
|
f"{config['base_url']}/v1/sessions",
|
|
headers=headers,
|
|
json=session_config,
|
|
timeout=30,
|
|
)
|
|
|
|
if response.status_code == 402 and enable_proxies:
|
|
proxies_fallback = True
|
|
logger.warning(
|
|
"Proxies unavailable (402), retrying without proxies. "
|
|
"Bot detection may be less effective."
|
|
)
|
|
session_config.pop("proxies", None)
|
|
response = requests.post(
|
|
f"{config['base_url']}/v1/sessions",
|
|
headers=headers,
|
|
json=session_config,
|
|
timeout=30,
|
|
)
|
|
|
|
if not response.ok:
|
|
raise RuntimeError(
|
|
f"Failed to create Browserbase session: "
|
|
f"{response.status_code} {response.text}"
|
|
)
|
|
|
|
session_data = response.json()
|
|
session_name = f"hermes_{task_id}_{uuid.uuid4().hex[:8]}"
|
|
|
|
if enable_proxies and not proxies_fallback:
|
|
features_enabled["proxies"] = True
|
|
if enable_advanced_stealth:
|
|
features_enabled["advanced_stealth"] = True
|
|
if enable_keep_alive and not keepalive_fallback:
|
|
features_enabled["keep_alive"] = True
|
|
if custom_timeout_ms and "timeout" in session_config:
|
|
features_enabled["custom_timeout"] = True
|
|
|
|
feature_str = ", ".join(k for k, v in features_enabled.items() if v)
|
|
logger.info("Created Browserbase session %s with features: %s", session_name, feature_str)
|
|
|
|
return {
|
|
"session_name": session_name,
|
|
"bb_session_id": session_data["id"],
|
|
"cdp_url": session_data["connectUrl"],
|
|
"features": features_enabled,
|
|
}
|
|
|
|
def close_session(self, session_id: str) -> bool:
|
|
try:
|
|
config = self._get_config()
|
|
except ValueError:
|
|
logger.warning("Cannot close Browserbase session %s — missing credentials", session_id)
|
|
return False
|
|
|
|
try:
|
|
response = requests.post(
|
|
f"{config['base_url']}/v1/sessions/{session_id}",
|
|
headers={
|
|
"X-BB-API-Key": config["api_key"],
|
|
"Content-Type": "application/json",
|
|
},
|
|
json={
|
|
"projectId": config["project_id"],
|
|
"status": "REQUEST_RELEASE",
|
|
},
|
|
timeout=10,
|
|
)
|
|
if response.status_code in (200, 201, 204):
|
|
logger.debug("Successfully closed Browserbase session %s", session_id)
|
|
return True
|
|
else:
|
|
logger.warning(
|
|
"Failed to close session %s: HTTP %s - %s",
|
|
session_id,
|
|
response.status_code,
|
|
response.text[:200],
|
|
)
|
|
return False
|
|
except Exception as e:
|
|
logger.error("Exception closing Browserbase session %s: %s", session_id, e)
|
|
return False
|
|
|
|
def emergency_cleanup(self, session_id: str) -> None:
|
|
config = self._get_config_or_none()
|
|
if config is None:
|
|
logger.warning("Cannot emergency-cleanup Browserbase session %s — missing credentials", session_id)
|
|
return
|
|
try:
|
|
requests.post(
|
|
f"{config['base_url']}/v1/sessions/{session_id}",
|
|
headers={
|
|
"X-BB-API-Key": config["api_key"],
|
|
"Content-Type": "application/json",
|
|
},
|
|
json={
|
|
"projectId": config["project_id"],
|
|
"status": "REQUEST_RELEASE",
|
|
},
|
|
timeout=5,
|
|
)
|
|
except Exception as e:
|
|
logger.debug("Emergency cleanup failed for Browserbase session %s: %s", session_id, e)
|