mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-25 00:51:20 +00:00
Production fixes: - voice_mode.py: add is_recording property to AudioRecorder (parity with TermuxAudioRecorder) - cronjob_tools.py: add sms example to deliver description Test fixes: - test_real_interrupt_subagent: add missing _execution_thread_id (fixes 19 cascading failures from leaked _build_system_prompt patch) - test_anthropic_error_handling: add _FakeMessages, override _interruptible_streaming_api_call (6 fixes) - test_ctx_halving_fix: add missing request_overrides attribute (4 fixes) - test_context_token_tracking: set _disable_streaming=True for non-streaming test path (4 fixes) - test_dict_tool_call_args: set _disable_streaming=True (1 fix) - test_provider_parity: add model='gpt-4o' for AIGateway tests to meet 64K minimum context (4 fixes) - test_session_race_guard: add user_id to SessionSource (5 fixes) - test_restart_drain/helpers: add user_id to SessionSource (2 fixes) - test_telegram_photo_interrupts: add user_id to SessionSource - test_interrupt: target thread_id for per-thread interrupt system (2 fixes) - test_zombie_process_cleanup: rewrite with object.__new__ for refactored GatewayRunner.stop() (1 fix) - test_browser_camofox_state: update config version 15->17 (1 fix) - test_trajectory_compressor_async: widen lookback window 10->20 for line-shifted AsyncOpenAI (1 fix) - test_voice_mode: fixed by production is_recording addition (5 fixes) - test_voice_cli_integration: add _attached_images to CLI stub (2 fixes) - test_hermes_logging: explicit propagation/level reset for cross-test pollution defense (1 fix) - test_run_agent: add base_url for OpenRouter detection tests (2 fixes) Deleted: - test_inline_think_blocks_reasoning_only_accepted: tested unimplemented inline <think> handling
73 lines
2 KiB
Python
73 lines
2 KiB
Python
import json
|
|
from types import SimpleNamespace
|
|
|
|
|
|
def _tool_call(name: str, arguments):
|
|
return SimpleNamespace(
|
|
id="call_1",
|
|
type="function",
|
|
function=SimpleNamespace(name=name, arguments=arguments),
|
|
)
|
|
|
|
|
|
def _response_with_tool_call(arguments):
|
|
assistant = SimpleNamespace(
|
|
content=None,
|
|
reasoning=None,
|
|
tool_calls=[_tool_call("read_file", arguments)],
|
|
)
|
|
choice = SimpleNamespace(message=assistant, finish_reason="tool_calls")
|
|
return SimpleNamespace(choices=[choice], usage=None)
|
|
|
|
|
|
class _FakeChatCompletions:
|
|
def __init__(self):
|
|
self.calls = 0
|
|
|
|
def create(self, **kwargs):
|
|
self.calls += 1
|
|
if self.calls == 1:
|
|
return _response_with_tool_call({"path": "README.md"})
|
|
return SimpleNamespace(
|
|
choices=[
|
|
SimpleNamespace(
|
|
message=SimpleNamespace(content="done", reasoning=None, tool_calls=[]),
|
|
finish_reason="stop",
|
|
)
|
|
],
|
|
usage=None,
|
|
)
|
|
|
|
|
|
class _FakeClient:
|
|
def __init__(self):
|
|
self.chat = SimpleNamespace(completions=_FakeChatCompletions())
|
|
|
|
|
|
def test_tool_call_validation_accepts_dict_arguments(monkeypatch):
|
|
from run_agent import AIAgent
|
|
|
|
monkeypatch.setattr("run_agent.OpenAI", lambda **kwargs: _FakeClient())
|
|
monkeypatch.setattr(
|
|
"run_agent.get_tool_definitions",
|
|
lambda *args, **kwargs: [{"function": {"name": "read_file"}}],
|
|
)
|
|
monkeypatch.setattr(
|
|
"run_agent.handle_function_call",
|
|
lambda name, args, task_id=None, **kwargs: json.dumps({"ok": True, "args": args}),
|
|
)
|
|
|
|
agent = AIAgent(
|
|
model="test-model",
|
|
api_key="test-key",
|
|
base_url="http://localhost:8080/v1",
|
|
platform="cli",
|
|
max_iterations=3,
|
|
quiet_mode=True,
|
|
skip_memory=True,
|
|
)
|
|
agent._disable_streaming = True
|
|
|
|
result = agent.run_conversation("read the file")
|
|
|
|
assert result["final_response"] == "done"
|