| """Tests for cloud browser provider runtime fallback to local Chromium. |
| |
| Covers the fallback logic in _get_session_info() when a cloud provider |
| is configured but fails at runtime (issue #10883). |
| """ |
| import logging |
| from unittest.mock import Mock, patch |
|
|
| import pytest |
|
|
| import tools.browser_tool as browser_tool |
|
|
|
|
| def _reset_session_state(monkeypatch): |
| """Clear caches so each test starts fresh.""" |
| monkeypatch.setattr(browser_tool, "_active_sessions", {}) |
| monkeypatch.setattr(browser_tool, "_cached_cloud_provider", None) |
| monkeypatch.setattr(browser_tool, "_cloud_provider_resolved", False) |
| monkeypatch.setattr(browser_tool, "_start_browser_cleanup_thread", lambda: None) |
| monkeypatch.setattr(browser_tool, "_update_session_activity", lambda t: None) |
|
|
|
|
| class TestCloudProviderRuntimeFallback: |
| """Tests for _get_session_info cloud → local fallback.""" |
|
|
| def test_cloud_failure_falls_back_to_local(self, monkeypatch): |
| """When cloud provider.create_session raises, fall back to local.""" |
| _reset_session_state(monkeypatch) |
|
|
| provider = Mock() |
| provider.create_session.side_effect = RuntimeError("401 Unauthorized") |
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None) |
|
|
| session = browser_tool._get_session_info("task-1") |
|
|
| assert session["fallback_from_cloud"] is True |
| assert "401 Unauthorized" in session["fallback_reason"] |
| assert session["fallback_provider"] == "Mock" |
| assert session["features"]["local"] is True |
| assert session["cdp_url"] is None |
|
|
| def test_cloud_success_no_fallback(self, monkeypatch): |
| """When cloud succeeds, no fallback markers are present.""" |
| _reset_session_state(monkeypatch) |
|
|
| provider = Mock() |
| provider.create_session.return_value = { |
| "session_name": "cloud-sess", |
| "bb_session_id": "bb_123", |
| "cdp_url": None, |
| "features": {"browser_use": True}, |
| } |
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None) |
|
|
| session = browser_tool._get_session_info("task-2") |
|
|
| assert session["session_name"] == "cloud-sess" |
| assert "fallback_from_cloud" not in session |
| assert "fallback_reason" not in session |
|
|
| def test_cloud_and_local_both_fail(self, monkeypatch): |
| """When both cloud and local fail, raise RuntimeError with both contexts.""" |
| _reset_session_state(monkeypatch) |
|
|
| provider = Mock() |
| provider.create_session.side_effect = RuntimeError("cloud boom") |
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None) |
| monkeypatch.setattr( |
| browser_tool, "_create_local_session", |
| Mock(side_effect=OSError("no chromium")), |
| ) |
|
|
| with pytest.raises(RuntimeError, match="cloud boom.*local.*no chromium"): |
| browser_tool._get_session_info("task-3") |
|
|
| def test_no_provider_uses_local_directly(self, monkeypatch): |
| """When no cloud provider is configured, local mode is used with no fallback markers.""" |
| _reset_session_state(monkeypatch) |
|
|
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: None) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None) |
|
|
| session = browser_tool._get_session_info("task-4") |
|
|
| assert session["features"]["local"] is True |
| assert "fallback_from_cloud" not in session |
|
|
| def test_cdp_override_bypasses_provider(self, monkeypatch): |
| """CDP override takes priority — cloud provider is never consulted.""" |
| _reset_session_state(monkeypatch) |
|
|
| provider = Mock() |
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: "ws://host:9222/devtools/browser/abc") |
|
|
| session = browser_tool._get_session_info("task-5") |
|
|
| provider.create_session.assert_not_called() |
| assert session["cdp_url"] == "ws://host:9222/devtools/browser/abc" |
|
|
| def test_fallback_logs_warning_with_provider_name(self, monkeypatch, caplog): |
| """Fallback emits a warning log with the provider class name and error.""" |
| _reset_session_state(monkeypatch) |
|
|
| BrowserUseProviderFake = type("BrowserUseProvider", (), { |
| "create_session": Mock(side_effect=ConnectionError("timeout")), |
| }) |
| provider = BrowserUseProviderFake() |
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None) |
|
|
| with caplog.at_level(logging.WARNING, logger="tools.browser_tool"): |
| session = browser_tool._get_session_info("task-6") |
|
|
| assert session["fallback_from_cloud"] is True |
| assert any("BrowserUseProvider" in r.message and "timeout" in r.message |
| for r in caplog.records) |
|
|
| def test_cloud_failure_does_not_poison_next_task(self, monkeypatch): |
| """A fallback for one task_id doesn't affect a new task_id when cloud recovers.""" |
| _reset_session_state(monkeypatch) |
|
|
| call_count = 0 |
|
|
| def create_session_flaky(task_id): |
| nonlocal call_count |
| call_count += 1 |
| if call_count == 1: |
| raise RuntimeError("transient failure") |
| return { |
| "session_name": "cloud-ok", |
| "bb_session_id": "bb_999", |
| "cdp_url": None, |
| "features": {"browser_use": True}, |
| } |
|
|
| provider = Mock() |
| provider.create_session.side_effect = create_session_flaky |
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None) |
|
|
| |
| s1 = browser_tool._get_session_info("task-a") |
| assert s1["fallback_from_cloud"] is True |
|
|
| |
| s2 = browser_tool._get_session_info("task-b") |
| assert "fallback_from_cloud" not in s2 |
| assert s2["session_name"] == "cloud-ok" |
|
|
| def test_cloud_returns_invalid_session_triggers_fallback(self, monkeypatch): |
| """Cloud provider returning None or empty dict triggers fallback.""" |
| _reset_session_state(monkeypatch) |
|
|
| provider = Mock() |
| provider.create_session.return_value = None |
| monkeypatch.setattr(browser_tool, "_get_cloud_provider", lambda: provider) |
| monkeypatch.setattr(browser_tool, "_get_cdp_override", lambda: None) |
|
|
| session = browser_tool._get_session_info("task-7") |
|
|
| assert session["fallback_from_cloud"] is True |
| assert "invalid session" in session["fallback_reason"] |
|
|