mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-15 09:21:36 +00:00
Custom endpoints carry two naming conventions for the same provider: the agent's provider attribute is the generic 'custom' label while the pool is keyed 'custom:<normalized-name>'. The defensive guard in recover_with_credential_pool compared them literally, logged 'Credential pool provider mismatch: pool=custom:<name>, agent=custom', and skipped recovery — so 401 refresh and 429 rotation never ran for ANY custom-provider user (seen in the field on a Fireworks setup whose dead key burned full retry cycles every turn with the skip warning on each one). Accept the pair only when the agent's CURRENT base_url resolves to the same pool key via get_custom_provider_pool_key, preserving the guard's original purpose (#33088/#33163): a fallback provider or a different custom endpoint still skips pool mutation.
103 lines
4 KiB
Python
103 lines
4 KiB
Python
"""Regression tests for the credential-pool provider-mismatch guard with
|
|
custom providers (Bernard's Fireworks report, June 2026).
|
|
|
|
Custom endpoints carry two naming conventions for the same provider: the
|
|
agent's ``provider`` attribute is the generic ``"custom"`` label while the
|
|
pool is keyed ``custom:<normalized-name>`` (``CUSTOM_POOL_PREFIX``). The
|
|
defensive guard in ``recover_with_credential_pool`` compared the two
|
|
literally, logged "Credential pool provider mismatch: pool=custom:<name>,
|
|
agent=custom", and skipped recovery — so 401/429 recovery (refresh,
|
|
rotation) never ran for ANY custom-provider user.
|
|
|
|
The fix accepts the pair only when the agent's current base_url resolves to
|
|
the same pool key, preserving the guard's original purpose (#33088/#33163:
|
|
never mutate the primary's pool while a fallback provider is active).
|
|
"""
|
|
from unittest.mock import MagicMock, patch
|
|
|
|
import pytest
|
|
|
|
from agent.agent_runtime_helpers import recover_with_credential_pool
|
|
from agent.error_classifier import FailoverReason
|
|
|
|
|
|
FIREWORKS_URL = "https://api.fireworks.ai/inference/v1"
|
|
|
|
|
|
def _agent(provider, base_url, pool_provider):
|
|
agent = MagicMock()
|
|
agent.provider = provider
|
|
agent.base_url = base_url
|
|
pool = MagicMock()
|
|
pool.provider = pool_provider
|
|
agent._credential_pool = pool
|
|
return agent, pool
|
|
|
|
|
|
class TestCustomPoolMismatchGuard:
|
|
def test_matching_custom_pool_reaches_recovery(self):
|
|
"""agent=custom + pool=custom:<name> whose base_url matches must NOT
|
|
be treated as a cross-provider mismatch."""
|
|
agent, pool = _agent("custom", FIREWORKS_URL, "custom:fireworks")
|
|
# Rate-limit path deterministically calls pool.current() once past
|
|
# the guard (the auth path consults agent._is_entitlement_failure,
|
|
# which a MagicMock would answer truthily).
|
|
pool.current.return_value = None
|
|
with patch(
|
|
"agent.credential_pool.get_custom_provider_pool_key",
|
|
return_value="custom:fireworks",
|
|
):
|
|
recover_with_credential_pool(
|
|
agent,
|
|
status_code=429,
|
|
has_retried_429=False,
|
|
classified_reason=FailoverReason.rate_limit,
|
|
)
|
|
assert pool.current.called, (
|
|
"guard short-circuited: pool never touched despite matching "
|
|
"custom base_url"
|
|
)
|
|
|
|
def test_unrelated_custom_pool_still_guarded(self):
|
|
"""agent=custom pointed at a DIFFERENT endpoint than the pool's
|
|
custom provider must still skip pool mutation."""
|
|
agent, pool = _agent(
|
|
"custom", "https://other-endpoint.example/v1", "custom:fireworks"
|
|
)
|
|
with patch(
|
|
"agent.credential_pool.get_custom_provider_pool_key",
|
|
return_value="custom:other",
|
|
):
|
|
recovered, _ = recover_with_credential_pool(
|
|
agent,
|
|
status_code=401,
|
|
has_retried_429=False,
|
|
classified_reason=FailoverReason.auth,
|
|
)
|
|
assert recovered is False
|
|
assert not pool.method_calls
|
|
|
|
def test_fallback_provider_still_guarded(self):
|
|
"""Original #33088/#33163 contract: when a fallback provider is
|
|
active (agent.provider != pool.provider, non-custom), the pool is
|
|
never mutated."""
|
|
agent, pool = _agent("openai-codex", "https://chatgpt.com/backend-api", "custom:fireworks")
|
|
recovered, _ = recover_with_credential_pool(
|
|
agent,
|
|
status_code=401,
|
|
has_retried_429=False,
|
|
classified_reason=FailoverReason.auth,
|
|
)
|
|
assert recovered is False
|
|
assert not pool.method_calls
|
|
|
|
def test_plain_provider_mismatch_still_guarded(self):
|
|
agent, pool = _agent("openrouter", "https://openrouter.ai/api/v1", "anthropic")
|
|
recovered, _ = recover_with_credential_pool(
|
|
agent,
|
|
status_code=429,
|
|
has_retried_429=False,
|
|
classified_reason=FailoverReason.rate_limit,
|
|
)
|
|
assert recovered is False
|
|
assert not pool.method_calls
|