mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-25 00:51:20 +00:00
The Copilot provider resolved context windows via models.dev static data, which does not include account-specific models (e.g. claude-opus-4.6-1m with 1M context). This adds the live Copilot /models API as a higher- priority source for copilot/copilot-acp/github-copilot providers. New helper get_copilot_model_context() in hermes_cli/models.py extracts capabilities.limits.max_prompt_tokens from the cached catalog. Results are cached in-process for 1 hour. In agent/model_metadata.py, step 5a queries the live API before falling through to models.dev (step 5b). This ensures account-specific models get correct context windows while standard models still have a fallback. Part 1 of #7731. Refs: #7272
134 lines
4.8 KiB
Python
134 lines
4.8 KiB
Python
"""Tests for Copilot live /models context-window resolution."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import time
|
|
from unittest.mock import patch
|
|
|
|
import pytest
|
|
|
|
from hermes_cli.models import get_copilot_model_context
|
|
|
|
|
|
# Sample catalog items mimicking the Copilot /models API response
|
|
_SAMPLE_CATALOG = [
|
|
{
|
|
"id": "claude-opus-4.6-1m",
|
|
"capabilities": {
|
|
"type": "chat",
|
|
"limits": {"max_prompt_tokens": 1000000, "max_output_tokens": 64000},
|
|
},
|
|
},
|
|
{
|
|
"id": "gpt-4.1",
|
|
"capabilities": {
|
|
"type": "chat",
|
|
"limits": {"max_prompt_tokens": 128000, "max_output_tokens": 32768},
|
|
},
|
|
},
|
|
{
|
|
"id": "claude-sonnet-4",
|
|
"capabilities": {
|
|
"type": "chat",
|
|
"limits": {"max_prompt_tokens": 200000, "max_output_tokens": 64000},
|
|
},
|
|
},
|
|
{
|
|
"id": "model-without-limits",
|
|
"capabilities": {"type": "chat"},
|
|
},
|
|
{
|
|
"id": "model-zero-limit",
|
|
"capabilities": {
|
|
"type": "chat",
|
|
"limits": {"max_prompt_tokens": 0},
|
|
},
|
|
},
|
|
]
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def _clear_cache():
|
|
"""Reset module-level cache before each test."""
|
|
import hermes_cli.models as mod
|
|
|
|
mod._copilot_context_cache = {}
|
|
mod._copilot_context_cache_time = 0.0
|
|
yield
|
|
mod._copilot_context_cache = {}
|
|
mod._copilot_context_cache_time = 0.0
|
|
|
|
|
|
class TestGetCopilotModelContext:
|
|
"""Tests for get_copilot_model_context()."""
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_returns_max_prompt_tokens(self, mock_fetch):
|
|
assert get_copilot_model_context("claude-opus-4.6-1m") == 1_000_000
|
|
assert get_copilot_model_context("gpt-4.1") == 128_000
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_returns_none_for_unknown_model(self, mock_fetch):
|
|
assert get_copilot_model_context("nonexistent-model") is None
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_skips_models_without_limits(self, mock_fetch):
|
|
assert get_copilot_model_context("model-without-limits") is None
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_skips_zero_limit(self, mock_fetch):
|
|
assert get_copilot_model_context("model-zero-limit") is None
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_caches_results(self, mock_fetch):
|
|
get_copilot_model_context("gpt-4.1")
|
|
get_copilot_model_context("claude-sonnet-4")
|
|
# Only one API call despite two lookups
|
|
assert mock_fetch.call_count == 1
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_cache_expires(self, mock_fetch):
|
|
import hermes_cli.models as mod
|
|
|
|
get_copilot_model_context("gpt-4.1")
|
|
assert mock_fetch.call_count == 1
|
|
|
|
# Expire the cache
|
|
mod._copilot_context_cache_time = time.time() - 7200
|
|
get_copilot_model_context("gpt-4.1")
|
|
assert mock_fetch.call_count == 2
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=None)
|
|
def test_returns_none_when_catalog_unavailable(self, mock_fetch):
|
|
assert get_copilot_model_context("gpt-4.1") is None
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=[])
|
|
def test_returns_none_for_empty_catalog(self, mock_fetch):
|
|
assert get_copilot_model_context("gpt-4.1") is None
|
|
|
|
|
|
class TestModelMetadataCopilotIntegration:
|
|
"""Test that get_model_context_length() uses Copilot live API for copilot provider."""
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_copilot_provider_uses_live_api(self, mock_fetch):
|
|
from agent.model_metadata import get_model_context_length
|
|
|
|
ctx = get_model_context_length("claude-opus-4.6-1m", provider="copilot")
|
|
assert ctx == 1_000_000
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=_SAMPLE_CATALOG)
|
|
def test_copilot_acp_provider_uses_live_api(self, mock_fetch):
|
|
from agent.model_metadata import get_model_context_length
|
|
|
|
ctx = get_model_context_length("claude-sonnet-4", provider="copilot-acp")
|
|
assert ctx == 200_000
|
|
|
|
@patch("hermes_cli.models.fetch_github_model_catalog", return_value=None)
|
|
def test_falls_through_when_catalog_unavailable(self, mock_fetch):
|
|
from agent.model_metadata import get_model_context_length
|
|
|
|
# Should not raise, should fall through to models.dev or defaults
|
|
ctx = get_model_context_length("gpt-4.1", provider="copilot")
|
|
assert isinstance(ctx, int)
|
|
assert ctx > 0
|