mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-25 00:51:20 +00:00
Automated dead code audit using vulture + coverage.py + ast-grep intersection, confirmed by Opus deep verification pass. Every symbol verified to have zero production callers (test imports excluded from reachability analysis). Removes ~1,534 lines of dead production code across 46 files and ~1,382 lines of stale test code. 3 entire files deleted (agent/builtin_memory_provider.py, hermes_cli/checklist.py, tests/hermes_cli/test_setup_model_selection.py). Co-authored-by: alt-glitch <balyan.sid@gmail.com>
202 lines
8.8 KiB
Python
202 lines
8.8 KiB
Python
"""Tests for hermes_cli.copilot_auth — Copilot token validation and resolution."""
|
|
|
|
import os
|
|
import pytest
|
|
from unittest.mock import patch, MagicMock
|
|
|
|
|
|
class TestTokenValidation:
|
|
"""Token type validation."""
|
|
|
|
def test_classic_pat_rejected(self):
|
|
from hermes_cli.copilot_auth import validate_copilot_token
|
|
valid, msg = validate_copilot_token("ghp_abcdefghijklmnop1234")
|
|
assert valid is False
|
|
assert "Classic Personal Access Tokens" in msg
|
|
assert "ghp_" in msg
|
|
|
|
def test_oauth_token_accepted(self):
|
|
from hermes_cli.copilot_auth import validate_copilot_token
|
|
valid, msg = validate_copilot_token("gho_abcdefghijklmnop1234")
|
|
assert valid is True
|
|
|
|
def test_fine_grained_pat_accepted(self):
|
|
from hermes_cli.copilot_auth import validate_copilot_token
|
|
valid, msg = validate_copilot_token("github_pat_abcdefghijklmnop1234")
|
|
assert valid is True
|
|
|
|
def test_github_app_token_accepted(self):
|
|
from hermes_cli.copilot_auth import validate_copilot_token
|
|
valid, msg = validate_copilot_token("ghu_abcdefghijklmnop1234")
|
|
assert valid is True
|
|
|
|
def test_empty_token_rejected(self):
|
|
from hermes_cli.copilot_auth import validate_copilot_token
|
|
valid, msg = validate_copilot_token("")
|
|
assert valid is False
|
|
|
|
|
|
|
|
class TestResolveToken:
|
|
"""Token resolution with env var priority."""
|
|
|
|
def test_copilot_github_token_first_priority(self, monkeypatch):
|
|
from hermes_cli.copilot_auth import resolve_copilot_token
|
|
monkeypatch.setenv("COPILOT_GITHUB_TOKEN", "gho_copilot_first")
|
|
monkeypatch.setenv("GH_TOKEN", "gho_gh_second")
|
|
monkeypatch.setenv("GITHUB_TOKEN", "gho_github_third")
|
|
token, source = resolve_copilot_token()
|
|
assert token == "gho_copilot_first"
|
|
assert source == "COPILOT_GITHUB_TOKEN"
|
|
|
|
def test_gh_token_second_priority(self, monkeypatch):
|
|
from hermes_cli.copilot_auth import resolve_copilot_token
|
|
monkeypatch.delenv("COPILOT_GITHUB_TOKEN", raising=False)
|
|
monkeypatch.setenv("GH_TOKEN", "gho_gh_second")
|
|
monkeypatch.setenv("GITHUB_TOKEN", "gho_github_third")
|
|
token, source = resolve_copilot_token()
|
|
assert token == "gho_gh_second"
|
|
assert source == "GH_TOKEN"
|
|
|
|
def test_github_token_third_priority(self, monkeypatch):
|
|
from hermes_cli.copilot_auth import resolve_copilot_token
|
|
monkeypatch.delenv("COPILOT_GITHUB_TOKEN", raising=False)
|
|
monkeypatch.delenv("GH_TOKEN", raising=False)
|
|
monkeypatch.setenv("GITHUB_TOKEN", "gho_github_third")
|
|
token, source = resolve_copilot_token()
|
|
assert token == "gho_github_third"
|
|
assert source == "GITHUB_TOKEN"
|
|
|
|
def test_classic_pat_in_env_skipped(self, monkeypatch):
|
|
"""Classic PATs in env vars should be skipped, not returned."""
|
|
from hermes_cli.copilot_auth import resolve_copilot_token
|
|
monkeypatch.setenv("COPILOT_GITHUB_TOKEN", "ghp_classic_pat_nope")
|
|
monkeypatch.delenv("GH_TOKEN", raising=False)
|
|
monkeypatch.setenv("GITHUB_TOKEN", "gho_valid_oauth")
|
|
token, source = resolve_copilot_token()
|
|
# Should skip the ghp_ token and find the gho_ one
|
|
assert token == "gho_valid_oauth"
|
|
assert source == "GITHUB_TOKEN"
|
|
|
|
def test_gh_cli_fallback(self, monkeypatch):
|
|
from hermes_cli.copilot_auth import resolve_copilot_token
|
|
monkeypatch.delenv("COPILOT_GITHUB_TOKEN", raising=False)
|
|
monkeypatch.delenv("GH_TOKEN", raising=False)
|
|
monkeypatch.delenv("GITHUB_TOKEN", raising=False)
|
|
with patch("hermes_cli.copilot_auth._try_gh_cli_token", return_value="gho_from_cli"):
|
|
token, source = resolve_copilot_token()
|
|
assert token == "gho_from_cli"
|
|
assert source == "gh auth token"
|
|
|
|
def test_gh_cli_classic_pat_raises(self, monkeypatch):
|
|
from hermes_cli.copilot_auth import resolve_copilot_token
|
|
monkeypatch.delenv("COPILOT_GITHUB_TOKEN", raising=False)
|
|
monkeypatch.delenv("GH_TOKEN", raising=False)
|
|
monkeypatch.delenv("GITHUB_TOKEN", raising=False)
|
|
with patch("hermes_cli.copilot_auth._try_gh_cli_token", return_value="ghp_classic"):
|
|
with pytest.raises(ValueError, match="classic PAT"):
|
|
resolve_copilot_token()
|
|
|
|
def test_no_token_returns_empty(self, monkeypatch):
|
|
from hermes_cli.copilot_auth import resolve_copilot_token
|
|
monkeypatch.delenv("COPILOT_GITHUB_TOKEN", raising=False)
|
|
monkeypatch.delenv("GH_TOKEN", raising=False)
|
|
monkeypatch.delenv("GITHUB_TOKEN", raising=False)
|
|
with patch("hermes_cli.copilot_auth._try_gh_cli_token", return_value=None):
|
|
token, source = resolve_copilot_token()
|
|
assert token == ""
|
|
assert source == ""
|
|
|
|
|
|
class TestRequestHeaders:
|
|
"""Copilot API header generation."""
|
|
|
|
def test_default_headers_include_openai_intent(self):
|
|
from hermes_cli.copilot_auth import copilot_request_headers
|
|
headers = copilot_request_headers()
|
|
assert headers["Openai-Intent"] == "conversation-edits"
|
|
assert headers["User-Agent"] == "HermesAgent/1.0"
|
|
assert "Editor-Version" in headers
|
|
|
|
def test_agent_turn_sets_initiator(self):
|
|
from hermes_cli.copilot_auth import copilot_request_headers
|
|
headers = copilot_request_headers(is_agent_turn=True)
|
|
assert headers["x-initiator"] == "agent"
|
|
|
|
def test_user_turn_sets_initiator(self):
|
|
from hermes_cli.copilot_auth import copilot_request_headers
|
|
headers = copilot_request_headers(is_agent_turn=False)
|
|
assert headers["x-initiator"] == "user"
|
|
|
|
def test_vision_header(self):
|
|
from hermes_cli.copilot_auth import copilot_request_headers
|
|
headers = copilot_request_headers(is_vision=True)
|
|
assert headers["Copilot-Vision-Request"] == "true"
|
|
|
|
def test_no_vision_header_by_default(self):
|
|
from hermes_cli.copilot_auth import copilot_request_headers
|
|
headers = copilot_request_headers()
|
|
assert "Copilot-Vision-Request" not in headers
|
|
|
|
|
|
class TestCopilotDefaultHeaders:
|
|
"""The models.py copilot_default_headers uses copilot_auth."""
|
|
|
|
def test_includes_openai_intent(self):
|
|
from hermes_cli.models import copilot_default_headers
|
|
headers = copilot_default_headers()
|
|
assert "Openai-Intent" in headers
|
|
assert headers["Openai-Intent"] == "conversation-edits"
|
|
|
|
def test_includes_x_initiator(self):
|
|
from hermes_cli.models import copilot_default_headers
|
|
headers = copilot_default_headers()
|
|
assert "x-initiator" in headers
|
|
|
|
|
|
class TestApiModeSelection:
|
|
"""API mode selection matching opencode's shouldUseCopilotResponsesApi."""
|
|
|
|
def test_gpt5_uses_responses(self):
|
|
from hermes_cli.models import _should_use_copilot_responses_api
|
|
assert _should_use_copilot_responses_api("gpt-5.4") is True
|
|
assert _should_use_copilot_responses_api("gpt-5.4-mini") is True
|
|
assert _should_use_copilot_responses_api("gpt-5.3-codex") is True
|
|
assert _should_use_copilot_responses_api("gpt-5.2-codex") is True
|
|
assert _should_use_copilot_responses_api("gpt-5.2") is True
|
|
assert _should_use_copilot_responses_api("gpt-5.1-codex-max") is True
|
|
|
|
def test_gpt5_mini_excluded(self):
|
|
from hermes_cli.models import _should_use_copilot_responses_api
|
|
assert _should_use_copilot_responses_api("gpt-5-mini") is False
|
|
|
|
def test_gpt4_uses_chat(self):
|
|
from hermes_cli.models import _should_use_copilot_responses_api
|
|
assert _should_use_copilot_responses_api("gpt-4.1") is False
|
|
assert _should_use_copilot_responses_api("gpt-4o") is False
|
|
assert _should_use_copilot_responses_api("gpt-4o-mini") is False
|
|
|
|
def test_non_gpt_uses_chat(self):
|
|
from hermes_cli.models import _should_use_copilot_responses_api
|
|
assert _should_use_copilot_responses_api("claude-sonnet-4.6") is False
|
|
assert _should_use_copilot_responses_api("claude-opus-4.6") is False
|
|
assert _should_use_copilot_responses_api("gemini-2.5-pro") is False
|
|
assert _should_use_copilot_responses_api("grok-code-fast-1") is False
|
|
|
|
|
|
class TestEnvVarOrder:
|
|
"""PROVIDER_REGISTRY has correct env var order."""
|
|
|
|
def test_copilot_env_vars_include_copilot_github_token(self):
|
|
from hermes_cli.auth import PROVIDER_REGISTRY
|
|
copilot = PROVIDER_REGISTRY["copilot"]
|
|
assert "COPILOT_GITHUB_TOKEN" in copilot.api_key_env_vars
|
|
# COPILOT_GITHUB_TOKEN should be first
|
|
assert copilot.api_key_env_vars[0] == "COPILOT_GITHUB_TOKEN"
|
|
|
|
def test_copilot_env_vars_order_matches_docs(self):
|
|
from hermes_cli.auth import PROVIDER_REGISTRY
|
|
copilot = PROVIDER_REGISTRY["copilot"]
|
|
assert copilot.api_key_env_vars == (
|
|
"COPILOT_GITHUB_TOKEN", "GH_TOKEN", "GITHUB_TOKEN"
|
|
)
|