hermes-agent/tests/hermes_cli/test_bedrock_model_picker.py
Ruda Porto Filgueiras 37551ee53e test(bedrock): add model picker and region routing tests
25 new tests (all Bedrock API calls mocked, no real AWS creds needed):

tests/hermes_cli/test_bedrock_model_picker.py (20 tests):
  - provider_model_ids("bedrock") uses live discovery, returns regional
    model IDs, falls back gracefully on empty/exception, resolves all
    bedrock aliases (aws, aws-bedrock, amazon-bedrock) to live discovery
  - list_authenticated_providers() section 2: bedrock appears with AWS
    creds, model list from discover_bedrock_models(), total_models
    matches, is_current flag works, absent creds hides bedrock, discovery
    failure does not crash, no duplicate entries
  - Region routing: botocore profile eu-central-1 yields eu.* model IDs
    end-to-end; env var takes priority over botocore profile
  - providers.py overlay: exists with correct transport/auth_type, label
    is non-empty, all aliases normalize to bedrock

tests/agent/test_bedrock_adapter.py (5 tests):
  - resolve_bedrock_region() botocore profile fallback, botocore failure
    fallback, us-east-1 hard fallback (with botocore mocked)
2026-04-28 03:53:11 -07:00

324 lines
16 KiB
Python

"""Tests for AWS Bedrock integration in the model picker and provider catalog.
Covers the three paths changed by fix/bedrock-provider-model-ids-live-discovery:
1. provider_model_ids("bedrock") — uses live discover_bedrock_models() instead
of the static _PROVIDER_MODELS table, with curated fallback.
2. list_authenticated_providers() Section 2 (HERMES_OVERLAYS) — bedrock
appears when AWS credentials are present; model list comes from live
discovery keyed by the resolved region, NOT the static us.* table.
3. Region resolution — resolve_bedrock_region() reads from botocore profile
when no AWS_REGION / AWS_DEFAULT_REGION env vars are set, so EU/AP users
in eu-central-1 get eu.* profile IDs, not us.* ones.
All Bedrock API calls are mocked — no real AWS credentials needed.
"""
import os
from unittest.mock import MagicMock, patch
import pytest
# ---------------------------------------------------------------------------
# Shared helpers / fixtures
# ---------------------------------------------------------------------------
_EU_MODELS = [
{"id": "eu.anthropic.claude-sonnet-4-6-20250514-v1:0", "name": "Claude Sonnet 4.6 (EU)", "provider": "inference-profile"},
{"id": "eu.anthropic.claude-haiku-4-5-20251015-v1:0", "name": "Claude Haiku 4.5 (EU)", "provider": "inference-profile"},
{"id": "eu.amazon.nova-pro-v1:0", "name": "Nova Pro (EU)", "provider": "inference-profile"},
]
_US_MODELS = [
{"id": "us.anthropic.claude-sonnet-4-6-20250514-v1:0", "name": "Claude Sonnet 4.6 (US)", "provider": "inference-profile"},
{"id": "us.amazon.nova-pro-v1:0", "name": "Nova Pro (US)", "provider": "inference-profile"},
]
def _mock_discover(region: str):
"""Return EU models for eu-* regions, US models otherwise."""
return _EU_MODELS if region.startswith("eu-") else _US_MODELS
# ---------------------------------------------------------------------------
# 1. provider_model_ids("bedrock")
# ---------------------------------------------------------------------------
class TestProviderModelIdsBedrock:
"""provider_model_ids("bedrock") should use live Bedrock discovery."""
def test_returns_live_discovered_model_ids(self, monkeypatch):
"""Live discovery result is returned as a flat list of model ID strings."""
from hermes_cli.models import provider_model_ids
monkeypatch.setenv("AWS_REGION", "eu-central-1")
with patch("agent.bedrock_adapter.discover_bedrock_models", side_effect=_mock_discover), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
result = provider_model_ids("bedrock")
assert "eu.anthropic.claude-sonnet-4-6-20250514-v1:0" in result
assert "eu.anthropic.claude-haiku-4-5-20251015-v1:0" in result
assert len(result) == len(_EU_MODELS)
def test_region_determines_model_ids(self, monkeypatch):
"""Different regions produce different model ID prefixes (eu.* vs us.*)."""
from hermes_cli.models import provider_model_ids
with patch("agent.bedrock_adapter.discover_bedrock_models", side_effect=_mock_discover):
with patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
eu_result = provider_model_ids("bedrock")
with patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="us-east-1"):
us_result = provider_model_ids("bedrock")
assert all(m.startswith("eu.") for m in eu_result)
assert all(m.startswith("us.") for m in us_result)
assert eu_result != us_result
def test_falls_back_to_static_list_when_discovery_empty(self, monkeypatch):
"""When discover_bedrock_models() returns [], fall back to curated static list."""
from hermes_cli.models import _PROVIDER_MODELS, provider_model_ids
with patch("agent.bedrock_adapter.discover_bedrock_models", return_value=[]), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
result = provider_model_ids("bedrock")
# Should fall back to static table (may be empty or populated depending on
# the current static list, but must not crash and must be a list).
assert isinstance(result, list)
def test_falls_back_to_static_list_on_exception(self, monkeypatch):
"""When discover_bedrock_models() raises, fall back gracefully."""
from hermes_cli.models import provider_model_ids
with patch("agent.bedrock_adapter.discover_bedrock_models",
side_effect=Exception("boto3 not installed")), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
result = provider_model_ids("bedrock")
assert isinstance(result, list) # no crash
def test_accepts_bedrock_aliases(self, monkeypatch):
"""Provider aliases (aws, aws-bedrock, amazon) should also trigger live discovery."""
from hermes_cli.models import provider_model_ids
_expected_ids = [m["id"] for m in _US_MODELS]
with patch("agent.bedrock_adapter.discover_bedrock_models", side_effect=_mock_discover), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="us-east-1"):
for alias in ("aws", "aws-bedrock", "amazon-bedrock"):
result = provider_model_ids(alias)
assert result == _expected_ids, \
f"alias {alias!r} should return live-discovered US model IDs, got {result!r}"
# ---------------------------------------------------------------------------
# 2. list_authenticated_providers() — bedrock via HERMES_OVERLAYS (Section 2)
# ---------------------------------------------------------------------------
class TestListAuthenticatedProvidersBedrock:
"""Bedrock should appear in the /model picker when AWS creds are present."""
def test_bedrock_appears_with_aws_profile(self, monkeypatch):
"""Bedrock shows up when AWS_PROFILE is set."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.setenv("AWS_PROFILE", "my-sso-profile")
monkeypatch.setenv("AWS_REGION", "eu-central-1")
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models", side_effect=_mock_discover), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
providers = list_authenticated_providers(current_provider="bedrock")
bedrock = next((p for p in providers if p["slug"] == "bedrock"), None)
assert bedrock is not None, "bedrock should appear when AWS credentials are present"
def test_bedrock_uses_live_discovery_not_static_list(self, monkeypatch):
"""Model IDs come from discover_bedrock_models(), not the static _PROVIDER_MODELS table."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.setenv("AWS_PROFILE", "my-sso-profile")
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models", side_effect=_mock_discover), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
providers = list_authenticated_providers(current_provider="bedrock")
bedrock = next((p for p in providers if p["slug"] == "bedrock"), None)
assert bedrock is not None
# All returned model IDs should have eu.* prefix — live discovery result
for model_id in bedrock["models"]:
assert model_id.startswith("eu."), \
f"Expected eu.* model ID from live discovery, got {model_id!r}"
def test_bedrock_total_models_matches_discovery(self, monkeypatch):
"""total_models reflects the actual discovered count."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.setenv("AWS_PROFILE", "my-sso-profile")
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models", return_value=_EU_MODELS), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
providers = list_authenticated_providers(current_provider="openai")
bedrock = next((p for p in providers if p["slug"] == "bedrock"), None)
assert bedrock is not None
assert bedrock["total_models"] == len(_EU_MODELS)
def test_bedrock_is_current_when_selected(self, monkeypatch):
"""is_current=True when current_provider matches bedrock."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.setenv("AWS_PROFILE", "my-sso-profile")
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models", return_value=_EU_MODELS), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
providers = list_authenticated_providers(current_provider="bedrock")
bedrock = next((p for p in providers if p["slug"] == "bedrock"), None)
assert bedrock is not None
assert bedrock["is_current"] is True
def test_bedrock_not_shown_without_credentials(self, monkeypatch):
"""Bedrock must not appear when no AWS credentials are present."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.delenv("AWS_PROFILE", raising=False)
monkeypatch.delenv("AWS_ACCESS_KEY_ID", raising=False)
monkeypatch.delenv("AWS_SECRET_ACCESS_KEY", raising=False)
monkeypatch.delenv("AWS_BEARER_TOKEN_BEDROCK", raising=False)
monkeypatch.delenv("AWS_WEB_IDENTITY_TOKEN_FILE", raising=False)
monkeypatch.delenv("AWS_CONTAINER_CREDENTIALS_RELATIVE_URI", raising=False)
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=False):
providers = list_authenticated_providers(current_provider="openai")
bedrock = next((p for p in providers if p["slug"] == "bedrock"), None)
assert bedrock is None, "bedrock should NOT appear when AWS credentials are absent"
def test_bedrock_falls_back_to_curated_when_discovery_fails(self, monkeypatch):
"""When discover_bedrock_models() raises, fall back to curated list without crashing."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.setenv("AWS_PROFILE", "my-sso-profile")
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models",
side_effect=Exception("API call failed")), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
providers = list_authenticated_providers(current_provider="bedrock")
# Should not raise — bedrock entry may or may not appear depending on
# whether the curated fallback has entries, but the call must succeed.
assert isinstance(providers, list)
def test_bedrock_no_duplicate_entries(self, monkeypatch):
"""Bedrock must appear at most once — not in both Section 1 and Section 2."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.setenv("AWS_PROFILE", "my-sso-profile")
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models", return_value=_EU_MODELS), \
patch("agent.bedrock_adapter.resolve_bedrock_region", return_value="eu-central-1"):
providers = list_authenticated_providers(current_provider="bedrock")
bedrock_entries = [p for p in providers if p["slug"] == "bedrock"]
assert len(bedrock_entries) <= 1, \
f"bedrock should appear at most once, got {len(bedrock_entries)} entries"
# ---------------------------------------------------------------------------
# 3. Region routing: EU/AP users see regional model IDs
# ---------------------------------------------------------------------------
class TestBedrockRegionRouting:
"""End-to-end: region from botocore profile is used for discovery, so EU/AP
users get eu.*/ap.* model IDs rather than the hardcoded us-east-1 list."""
def test_eu_region_from_botocore_profile_yields_eu_models(self):
"""When botocore resolves eu-central-1, picker shows eu.* model IDs."""
from hermes_cli.model_switch import list_authenticated_providers
mock_session = MagicMock()
mock_session.get_config_variable.return_value = "eu-central-1"
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models", side_effect=_mock_discover), \
patch("botocore.session.get_session", return_value=mock_session):
providers = list_authenticated_providers(current_provider="bedrock")
bedrock = next((p for p in providers if p["slug"] == "bedrock"), None)
assert bedrock is not None
for model_id in bedrock["models"]:
assert model_id.startswith("eu."), \
f"Expected eu.* model ID from eu-central-1 profile, got {model_id!r}"
def test_us_region_from_env_var_yields_us_models(self, monkeypatch):
"""Explicit AWS_REGION=us-east-1 returns us.* model IDs."""
from hermes_cli.model_switch import list_authenticated_providers
monkeypatch.setenv("AWS_REGION", "us-east-1")
with patch("agent.bedrock_adapter.has_aws_credentials", return_value=True), \
patch("agent.bedrock_adapter.discover_bedrock_models", side_effect=_mock_discover):
providers = list_authenticated_providers(current_provider="bedrock")
bedrock = next((p for p in providers if p["slug"] == "bedrock"), None)
assert bedrock is not None
for model_id in bedrock["models"]:
assert model_id.startswith("us."), \
f"Expected us.* model ID from us-east-1, got {model_id!r}"
def test_env_var_takes_priority_over_botocore_profile(self, monkeypatch):
"""AWS_REGION env var wins over botocore profile region."""
from agent.bedrock_adapter import resolve_bedrock_region
monkeypatch.setenv("AWS_REGION", "us-west-2")
mock_session = MagicMock()
mock_session.get_config_variable.return_value = "eu-central-1"
with patch("botocore.session.get_session", return_value=mock_session):
region = resolve_bedrock_region()
assert region == "us-west-2", "env var should override botocore profile"
# ---------------------------------------------------------------------------
# 4. providers.py overlay registration
# ---------------------------------------------------------------------------
class TestBedrockOverlayRegistration:
"""bedrock entry in HERMES_OVERLAYS is correctly configured."""
def test_bedrock_overlay_exists(self):
from hermes_cli.providers import HERMES_OVERLAYS
assert "bedrock" in HERMES_OVERLAYS
def test_bedrock_overlay_transport(self):
from hermes_cli.providers import HERMES_OVERLAYS
assert HERMES_OVERLAYS["bedrock"].transport == "bedrock_converse"
def test_bedrock_overlay_auth_type(self):
from hermes_cli.providers import HERMES_OVERLAYS
assert HERMES_OVERLAYS["bedrock"].auth_type == "aws_sdk"
def test_bedrock_label(self):
from hermes_cli.providers import get_label
label = get_label("bedrock")
assert label # non-empty
assert "bedrock" in label.lower() or "aws" in label.lower()
def test_bedrock_aliases_resolve(self):
from hermes_cli.providers import normalize_provider
for alias in ("aws", "aws-bedrock", "amazon-bedrock", "amazon"):
assert normalize_provider(alias) == "bedrock", \
f"alias {alias!r} should normalize to 'bedrock'"