mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-18 04:41:56 +00:00
125 lines
4.7 KiB
Python
125 lines
4.7 KiB
Python
"""Tests for ``_get_cloud_provider()`` caching policy.
|
|
|
|
Regression coverage for issue #22324: a transient ``None`` from the resolver
|
|
must not be cached for the lifetime of the process. Cache only when:
|
|
|
|
* The user explicitly opts in to ``cloud_provider: local``, OR
|
|
* A provider is successfully resolved.
|
|
|
|
All other ``None`` outcomes (no credentials yet, config read error, explicit
|
|
provider instantiation failure) leave the cache unset so the next call retries.
|
|
"""
|
|
import logging
|
|
from unittest.mock import Mock
|
|
|
|
import pytest
|
|
|
|
import tools.browser_tool as browser_tool
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def _reset_resolver_state(monkeypatch):
|
|
monkeypatch.setattr(browser_tool, "_cached_cloud_provider", None)
|
|
monkeypatch.setattr(browser_tool, "_cloud_provider_resolved", False)
|
|
yield
|
|
|
|
|
|
class TestCloudProviderCachePolicy:
|
|
def test_explicit_local_caches_permanently(self, monkeypatch):
|
|
"""`cloud_provider: local` is a positive choice and must stick."""
|
|
monkeypatch.setattr(
|
|
"hermes_cli.config.read_raw_config",
|
|
lambda: {"browser": {"cloud_provider": "local"}},
|
|
)
|
|
|
|
assert browser_tool._get_cloud_provider() is None
|
|
assert browser_tool._cloud_provider_resolved is True
|
|
|
|
# Even if config later changes, the cache stays.
|
|
monkeypatch.setattr(
|
|
"hermes_cli.config.read_raw_config",
|
|
lambda: {"browser": {"cloud_provider": "browser-use"}},
|
|
)
|
|
assert browser_tool._get_cloud_provider() is None
|
|
|
|
def test_successful_cloud_resolution_caches_permanently(self, monkeypatch):
|
|
"""A real provider instance must be cached and reused."""
|
|
fake_provider = Mock(name="BrowserUseProvider-instance")
|
|
factory = Mock(return_value=fake_provider)
|
|
monkeypatch.setattr(
|
|
browser_tool, "_PROVIDER_REGISTRY", {"browser-use": factory}
|
|
)
|
|
monkeypatch.setattr(
|
|
"hermes_cli.config.read_raw_config",
|
|
lambda: {"browser": {"cloud_provider": "browser-use"}},
|
|
)
|
|
|
|
assert browser_tool._get_cloud_provider() is fake_provider
|
|
assert browser_tool._cloud_provider_resolved is True
|
|
|
|
# Subsequent calls hit the cache; factory not called again.
|
|
assert browser_tool._get_cloud_provider() is fake_provider
|
|
assert factory.call_count == 1
|
|
|
|
def test_no_credentials_yet_does_not_cache_none(self, monkeypatch):
|
|
"""Auto-detect path with no creds: must NOT poison the cache."""
|
|
monkeypatch.setattr(
|
|
"hermes_cli.config.read_raw_config",
|
|
lambda: {"browser": {}},
|
|
)
|
|
|
|
bu_unconfigured = Mock()
|
|
bu_unconfigured.is_configured.return_value = False
|
|
bb_unconfigured = Mock()
|
|
bb_unconfigured.is_configured.return_value = False
|
|
monkeypatch.setattr(
|
|
browser_tool, "BrowserUseProvider", lambda: bu_unconfigured
|
|
)
|
|
monkeypatch.setattr(
|
|
browser_tool, "BrowserbaseProvider", lambda: bb_unconfigured
|
|
)
|
|
|
|
assert browser_tool._get_cloud_provider() is None
|
|
assert browser_tool._cloud_provider_resolved is False
|
|
|
|
# Credentials self-heal — next call must retry and pick up the provider.
|
|
healed = Mock(name="healed-provider")
|
|
healed.is_configured.return_value = True
|
|
monkeypatch.setattr(browser_tool, "BrowserUseProvider", lambda: healed)
|
|
|
|
assert browser_tool._get_cloud_provider() is healed
|
|
assert browser_tool._cloud_provider_resolved is True
|
|
|
|
def test_config_read_failure_does_not_cache_none(self, monkeypatch):
|
|
"""A raised config read must not pin the resolver to local mode."""
|
|
def boom():
|
|
raise OSError("config file locked")
|
|
|
|
monkeypatch.setattr("hermes_cli.config.read_raw_config", boom)
|
|
|
|
assert browser_tool._get_cloud_provider() is None
|
|
assert browser_tool._cloud_provider_resolved is False
|
|
|
|
def test_explicit_provider_instantiation_failure_does_not_cache(
|
|
self, monkeypatch, caplog
|
|
):
|
|
"""If `_PROVIDER_REGISTRY[key]()` raises, log warning and don't cache."""
|
|
def exploding_factory():
|
|
raise RuntimeError("missing dependency")
|
|
|
|
monkeypatch.setattr(
|
|
browser_tool, "_PROVIDER_REGISTRY", {"browser-use": exploding_factory}
|
|
)
|
|
monkeypatch.setattr(
|
|
"hermes_cli.config.read_raw_config",
|
|
lambda: {"browser": {"cloud_provider": "browser-use"}},
|
|
)
|
|
|
|
with caplog.at_level(logging.WARNING, logger="tools.browser_tool"):
|
|
assert browser_tool._get_cloud_provider() is None
|
|
|
|
assert browser_tool._cloud_provider_resolved is False
|
|
assert any(
|
|
"browser-use" in r.message and r.levelno == logging.WARNING
|
|
for r in caplog.records
|
|
)
|