"""Tests for GatewayRunner._format_session_info — session config surfacing.""" import pytest from unittest.mock import patch, MagicMock from pathlib import Path from gateway.run import GatewayRunner @pytest.fixture() def runner(): """Create a bare GatewayRunner without __init__.""" return GatewayRunner.__new__(GatewayRunner) def _patch_info(tmp_path, config_yaml, model, runtime): """Return a context-manager stack that patches _format_session_info deps.""" cfg_path = tmp_path / "config.yaml" if config_yaml is not None: cfg_path.write_text(config_yaml) return ( patch("gateway.run._hermes_home", tmp_path), patch("gateway.run._resolve_gateway_model", return_value=model), patch("gateway.run._resolve_runtime_agent_kwargs", return_value=runtime), ) class TestFormatSessionInfo: def test_includes_model_name(self, runner, tmp_path): p1, p2, p3 = _patch_info(tmp_path, "model:\n default: anthropic/claude-opus-4.6\n provider: openrouter\n", "anthropic/claude-opus-4.6", {"provider": "openrouter", "base_url": "https://openrouter.ai/api/v1", "api_key": "k"}) with p1, p2, p3: info = runner._format_session_info() assert "claude-opus-4.6" in info def test_includes_provider(self, runner, tmp_path): p1, p2, p3 = _patch_info(tmp_path, "model:\n default: test-model\n provider: openrouter\n", "test-model", {"provider": "openrouter", "base_url": "", "api_key": ""}) with p1, p2, p3: info = runner._format_session_info() assert "openrouter" in info def test_config_context_length(self, runner, tmp_path): p1, p2, p3 = _patch_info(tmp_path, "model:\n default: test-model\n context_length: 32768\n", "test-model", {"provider": "custom", "base_url": "", "api_key": ""}) with p1, p2, p3: info = runner._format_session_info() assert "32K" in info assert "config" in info def test_default_fallback_hint(self, runner, tmp_path): p1, p2, p3 = _patch_info(tmp_path, "model:\n default: unknown-model-xyz\n", "unknown-model-xyz", {"provider": "", "base_url": "", "api_key": ""}) with p1, p2, p3: info = runner._format_session_info() assert "128K" in info assert "model.context_length" in info def test_local_endpoint_shown(self, runner, tmp_path): p1, p2, p3 = _patch_info( tmp_path, "model:\n default: qwen3:8b\n provider: custom\n base_url: http://localhost:11434/v1\n context_length: 8192\n", "qwen3:8b", {"provider": "custom", "base_url": "http://localhost:11434/v1", "api_key": ""}) with p1, p2, p3: info = runner._format_session_info() assert "localhost:11434" in info assert "8K" in info def test_cloud_endpoint_hidden(self, runner, tmp_path): p1, p2, p3 = _patch_info(tmp_path, "model:\n default: test-model\n provider: openrouter\n", "test-model", {"provider": "openrouter", "base_url": "https://openrouter.ai/api/v1", "api_key": "k"}) with p1, p2, p3: info = runner._format_session_info() assert "Endpoint" not in info def test_million_context_format(self, runner, tmp_path): p1, p2, p3 = _patch_info(tmp_path, "model:\n default: test-model\n context_length: 1000000\n", "test-model", {"provider": "", "base_url": "", "api_key": ""}) with p1, p2, p3: info = runner._format_session_info() assert "1.0M" in info def test_missing_config(self, runner, tmp_path): """No config.yaml should not crash.""" p1, p2, p3 = _patch_info(tmp_path, None, # don't create config "anthropic/claude-sonnet-4.6", {"provider": "openrouter", "base_url": "", "api_key": ""}) with p1, p2, p3: info = runner._format_session_info() assert "Model" in info assert "Context" in info def test_runtime_resolution_failure_doesnt_crash(self, runner, tmp_path): """If runtime resolution raises, should still produce output.""" cfg_path = tmp_path / "config.yaml" cfg_path.write_text("model:\n default: test-model\n context_length: 4096\n") with patch("gateway.run._hermes_home", tmp_path), \ patch("gateway.run._resolve_gateway_model", return_value="test-model"), \ patch("gateway.run._resolve_runtime_agent_kwargs", side_effect=RuntimeError("no creds")): info = runner._format_session_info() assert "4K" in info assert "config" in info