mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-01 08:21:50 +08:00
249 lines
11 KiB
Python
249 lines
11 KiB
Python
|
|
"""Tests for hybrid browser-backend routing (LAN/localhost auto-local).
|
||
|
|
|
||
|
|
When a cloud browser provider (Browserbase / Browser-Use / Firecrawl) is
|
||
|
|
configured globally, ``browser.auto_local_for_private_urls`` (default True)
|
||
|
|
causes ``browser_navigate`` to transparently spawn a local Chromium sidecar
|
||
|
|
for URLs whose host resolves to a private/loopback/LAN address, while
|
||
|
|
public URLs continue to hit the cloud session in the same conversation.
|
||
|
|
|
||
|
|
These tests cover the routing decision layer — session_key selection,
|
||
|
|
sidecar detection, last-active-session tracking, and the config toggle.
|
||
|
|
The downstream session creation is covered by test_browser_cloud_fallback.py.
|
||
|
|
"""
|
||
|
|
from unittest.mock import Mock
|
||
|
|
|
||
|
|
import pytest
|
||
|
|
|
||
|
|
import tools.browser_tool as browser_tool
|
||
|
|
|
||
|
|
|
||
|
|
@pytest.fixture(autouse=True)
|
||
|
|
def _reset_routing_state(monkeypatch):
|
||
|
|
"""Clear module-level caches so each test starts clean."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_active_sessions", {})
|
||
|
|
monkeypatch.setattr(browser_tool, "_last_active_session_key", {})
|
||
|
|
monkeypatch.setattr(browser_tool, "_cached_cloud_provider", None)
|
||
|
|
monkeypatch.setattr(browser_tool, "_cloud_provider_resolved", False)
|
||
|
|
monkeypatch.setattr(browser_tool, "_auto_local_for_private_urls_resolved", False)
|
||
|
|
monkeypatch.setattr(browser_tool, "_cached_auto_local_for_private_urls", True)
|
||
|
|
monkeypatch.setattr(browser_tool, "_start_browser_cleanup_thread", lambda: None)
|
||
|
|
monkeypatch.setattr(browser_tool, "_update_session_activity", lambda t: None)
|
||
|
|
# Default: no CDP override, no Camofox
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None)
|
||
|
|
monkeypatch.setattr(browser_tool, "_is_camofox_mode", lambda: False)
|
||
|
|
|
||
|
|
|
||
|
|
class TestNavigationSessionKey:
|
||
|
|
"""Tests for _navigation_session_key URL-based routing decisions."""
|
||
|
|
|
||
|
|
def test_public_url_uses_bare_task_id(self, monkeypatch):
|
||
|
|
"""Public URL with cloud provider configured → bare task_id (cloud)."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
key = browser_tool._navigation_session_key("default", "https://github.com/x/y")
|
||
|
|
assert key == "default"
|
||
|
|
|
||
|
|
def test_localhost_routes_to_local_sidecar(self, monkeypatch):
|
||
|
|
"""``localhost`` URL → ``::local`` suffix when cloud configured + flag on."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://localhost:3000/")
|
||
|
|
assert key == "default::local"
|
||
|
|
|
||
|
|
def test_loopback_ipv4_routes_to_local_sidecar(self, monkeypatch):
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://127.0.0.1:8080/")
|
||
|
|
assert key == "default::local"
|
||
|
|
|
||
|
|
def test_rfc1918_lan_routes_to_local_sidecar(self, monkeypatch):
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://192.168.1.50:8000/")
|
||
|
|
assert key == "default::local"
|
||
|
|
|
||
|
|
def test_ipv6_loopback_routes_to_local_sidecar(self, monkeypatch):
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://[::1]:3000/")
|
||
|
|
assert key == "default::local"
|
||
|
|
|
||
|
|
def test_public_ip_literal_uses_bare_task_id(self, monkeypatch):
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
key = browser_tool._navigation_session_key("default", "https://8.8.8.8/")
|
||
|
|
assert key == "default"
|
||
|
|
|
||
|
|
def test_mdns_local_hostname_routes_to_sidecar(self, monkeypatch):
|
||
|
|
"""``*.local`` mDNS / ``*.lan`` / ``*.internal`` hostnames route to sidecar."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
for host in ("raspberrypi.local", "printer.lan", "db.internal"):
|
||
|
|
key = browser_tool._navigation_session_key("default", f"http://{host}/")
|
||
|
|
assert key == "default::local", f"host {host!r} did not route to sidecar"
|
||
|
|
|
||
|
|
def test_no_cloud_provider_stays_on_bare_task_id(self, monkeypatch):
|
||
|
|
"""When cloud provider is not configured, no hybrid routing happens."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: None)
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://localhost:3000/")
|
||
|
|
assert key == "default"
|
||
|
|
|
||
|
|
def test_camofox_mode_stays_on_bare_task_id(self, monkeypatch):
|
||
|
|
"""Camofox is already local — no hybrid routing needed."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
monkeypatch.setattr(browser_tool, "_is_camofox_mode", lambda: True)
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://localhost:3000/")
|
||
|
|
assert key == "default"
|
||
|
|
|
||
|
|
def test_cdp_override_stays_on_bare_task_id(self, monkeypatch):
|
||
|
|
"""A user-supplied CDP endpoint owns the whole session — no hybrid."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: "ws://localhost:9222")
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://localhost:3000/")
|
||
|
|
assert key == "default"
|
||
|
|
|
||
|
|
def test_feature_flag_off_disables_hybrid_routing(self, monkeypatch):
|
||
|
|
"""``auto_local_for_private_urls: false`` keeps private URLs on cloud."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
monkeypatch.setattr(browser_tool, "_auto_local_for_private_urls", lambda: False)
|
||
|
|
key = browser_tool._navigation_session_key("default", "http://localhost:3000/")
|
||
|
|
assert key == "default"
|
||
|
|
|
||
|
|
def test_none_task_id_defaults(self, monkeypatch):
|
||
|
|
"""``None`` task_id resolves to 'default'."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: Mock())
|
||
|
|
key = browser_tool._navigation_session_key(None, "http://localhost:3000/")
|
||
|
|
assert key == "default::local"
|
||
|
|
|
||
|
|
|
||
|
|
class TestSessionKeyHelpers:
|
||
|
|
def test_is_local_sidecar_key(self):
|
||
|
|
assert browser_tool._is_local_sidecar_key("default::local")
|
||
|
|
assert browser_tool._is_local_sidecar_key("my_task::local")
|
||
|
|
assert not browser_tool._is_local_sidecar_key("default")
|
||
|
|
assert not browser_tool._is_local_sidecar_key("my_task")
|
||
|
|
|
||
|
|
def test_last_session_key_falls_back_to_task_id(self, monkeypatch):
|
||
|
|
"""Without a recorded last-active key, returns the bare task_id."""
|
||
|
|
monkeypatch.setattr(browser_tool, "_last_active_session_key", {})
|
||
|
|
assert browser_tool._last_session_key("default") == "default"
|
||
|
|
assert browser_tool._last_session_key("task-42") == "task-42"
|
||
|
|
assert browser_tool._last_session_key(None) == "default"
|
||
|
|
|
||
|
|
def test_last_session_key_returns_recorded_key(self, monkeypatch):
|
||
|
|
monkeypatch.setattr(
|
||
|
|
browser_tool,
|
||
|
|
"_last_active_session_key",
|
||
|
|
{"default": "default::local", "task-42": "task-42"},
|
||
|
|
)
|
||
|
|
assert browser_tool._last_session_key("default") == "default::local"
|
||
|
|
assert browser_tool._last_session_key("task-42") == "task-42"
|
||
|
|
# Unknown task_id still falls back
|
||
|
|
assert browser_tool._last_session_key("other") == "other"
|
||
|
|
|
||
|
|
|
||
|
|
class TestHybridRoutingSessionCreation:
|
||
|
|
"""_get_session_info must force a local session when the key carries ``::local``."""
|
||
|
|
|
||
|
|
def test_local_sidecar_key_skips_cloud_provider(self, monkeypatch):
|
||
|
|
"""A ``::local``-suffixed key creates a local session even when cloud is set."""
|
||
|
|
provider = Mock()
|
||
|
|
provider.create_session.return_value = {
|
||
|
|
"session_name": "should_not_be_used",
|
||
|
|
"bb_session_id": "bb_xxx",
|
||
|
|
"cdp_url": "wss://fake.browserbase.com/ws",
|
||
|
|
}
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider)
|
||
|
|
monkeypatch.setattr(browser_tool, "_ensure_cdp_supervisor", lambda t: None)
|
||
|
|
|
||
|
|
session = browser_tool._get_session_info("default::local")
|
||
|
|
|
||
|
|
assert provider.create_session.call_count == 0
|
||
|
|
assert session["bb_session_id"] is None
|
||
|
|
assert session["cdp_url"] is None
|
||
|
|
assert session["features"]["local"] is True
|
||
|
|
|
||
|
|
def test_bare_task_id_with_cloud_provider_uses_cloud(self, monkeypatch):
|
||
|
|
"""A bare task_id with cloud provider configured hits the cloud path."""
|
||
|
|
provider = Mock()
|
||
|
|
provider.create_session.return_value = {
|
||
|
|
"session_name": "cloud-sess",
|
||
|
|
"bb_session_id": "bb_123",
|
||
|
|
"cdp_url": "wss://real.browserbase.com/ws",
|
||
|
|
}
|
||
|
|
monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider)
|
||
|
|
monkeypatch.setattr(browser_tool, "_ensure_cdp_supervisor", lambda t: None)
|
||
|
|
monkeypatch.setattr(browser_tool, "_resolve_cdp_override", lambda u: u)
|
||
|
|
|
||
|
|
session = browser_tool._get_session_info("default")
|
||
|
|
|
||
|
|
assert provider.create_session.call_count == 1
|
||
|
|
assert session["bb_session_id"] == "bb_123"
|
||
|
|
|
||
|
|
|
||
|
|
class TestCleanupHybridSessions:
|
||
|
|
"""cleanup_browser(bare_task_id) must reap both cloud + local sidecar sessions."""
|
||
|
|
|
||
|
|
def test_cleanup_reaps_both_primary_and_sidecar(self, monkeypatch):
|
||
|
|
"""Given a bare task_id with both sessions alive, both get cleaned."""
|
||
|
|
reaped = []
|
||
|
|
|
||
|
|
def _fake_cleanup_one(key):
|
||
|
|
reaped.append(key)
|
||
|
|
|
||
|
|
monkeypatch.setattr(browser_tool, "_cleanup_single_browser_session", _fake_cleanup_one)
|
||
|
|
monkeypatch.setattr(
|
||
|
|
browser_tool,
|
||
|
|
"_active_sessions",
|
||
|
|
{
|
||
|
|
"default": {"session_name": "cloud_sess"},
|
||
|
|
"default::local": {"session_name": "local_sess"},
|
||
|
|
},
|
||
|
|
)
|
||
|
|
monkeypatch.setattr(
|
||
|
|
browser_tool, "_last_active_session_key", {"default": "default::local"}
|
||
|
|
)
|
||
|
|
|
||
|
|
browser_tool.cleanup_browser("default")
|
||
|
|
|
||
|
|
assert set(reaped) == {"default", "default::local"}
|
||
|
|
# last-active pointer dropped
|
||
|
|
assert "default" not in browser_tool._last_active_session_key
|
||
|
|
|
||
|
|
def test_cleanup_reaps_only_primary_when_no_sidecar(self, monkeypatch):
|
||
|
|
"""When no sidecar exists, only the primary is reaped."""
|
||
|
|
reaped = []
|
||
|
|
|
||
|
|
def _fake_cleanup_one(key):
|
||
|
|
reaped.append(key)
|
||
|
|
|
||
|
|
monkeypatch.setattr(browser_tool, "_cleanup_single_browser_session", _fake_cleanup_one)
|
||
|
|
monkeypatch.setattr(
|
||
|
|
browser_tool,
|
||
|
|
"_active_sessions",
|
||
|
|
{"default": {"session_name": "cloud_sess"}},
|
||
|
|
)
|
||
|
|
|
||
|
|
browser_tool.cleanup_browser("default")
|
||
|
|
|
||
|
|
assert reaped == ["default"]
|
||
|
|
|
||
|
|
def test_cleanup_sidecar_directly_keeps_primary(self, monkeypatch):
|
||
|
|
"""Calling cleanup with a ``::local`` key reaps only the sidecar."""
|
||
|
|
reaped = []
|
||
|
|
|
||
|
|
def _fake_cleanup_one(key):
|
||
|
|
reaped.append(key)
|
||
|
|
|
||
|
|
monkeypatch.setattr(browser_tool, "_cleanup_single_browser_session", _fake_cleanup_one)
|
||
|
|
monkeypatch.setattr(
|
||
|
|
browser_tool,
|
||
|
|
"_active_sessions",
|
||
|
|
{
|
||
|
|
"default": {"session_name": "cloud_sess"},
|
||
|
|
"default::local": {"session_name": "local_sess"},
|
||
|
|
},
|
||
|
|
)
|
||
|
|
monkeypatch.setattr(
|
||
|
|
browser_tool, "_last_active_session_key", {"default": "default::local"}
|
||
|
|
)
|
||
|
|
|
||
|
|
browser_tool.cleanup_browser("default::local")
|
||
|
|
|
||
|
|
assert reaped == ["default::local"]
|
||
|
|
# Last-active pointer NOT dropped (primary task is still alive)
|
||
|
|
assert browser_tool._last_active_session_key.get("default") == "default::local"
|