mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-06 10:47:12 +08:00
Gemini's OpenAI-compatibility endpoint strictly requires the `name` field
on `role: tool` messages — it returns HTTP 400 ("Request contains an
invalid argument") when the function name is missing. OpenAI/Anthropic/
ollama tolerate the absence, so the gap stays invisible until the
conversation accumulates a tool turn and the user routes it through Gemini
(direct API or via ollama-cloud proxy).
Fix: add a `_get_tool_call_name_static()` helper alongside the existing
`_get_tool_call_id_static()`, and populate `name` at every site that
constructs a `role: tool` message — the pre-call sanitizer stub, the
tool-call args repair marker, both interrupt-skip paths, both
result-append paths (parallel + sequential), the invalid-tool-name
recovery, the invalid-JSON-args recovery, and the exception fallback.
Each call site was already in scope of the function name (`function_name`,
`skipped_name`, `name`, or a dict tool_call), so the change is local —
no new lookups, no behavior change for providers that already worked.
Fixes #16478
159 lines
4.6 KiB
Python
159 lines
4.6 KiB
Python
"""Tests for AIAgent._sanitize_tool_call_arguments."""
|
|
|
|
import copy
|
|
import logging
|
|
|
|
from run_agent import AIAgent
|
|
|
|
|
|
_MISSING = object()
|
|
|
|
|
|
def _tool_call(call_id="call_1", name="read_file", arguments='{"path":"/tmp/foo"}'):
|
|
function = {"name": name}
|
|
if arguments is not _MISSING:
|
|
function["arguments"] = arguments
|
|
return {
|
|
"id": call_id,
|
|
"type": "function",
|
|
"function": function,
|
|
}
|
|
|
|
|
|
def _assistant_message(*tool_calls):
|
|
return {
|
|
"role": "assistant",
|
|
"content": "tooling",
|
|
"tool_calls": list(tool_calls),
|
|
}
|
|
|
|
|
|
def _tool_message(call_id="call_1", content="ok"):
|
|
return {
|
|
"role": "tool",
|
|
"tool_call_id": call_id,
|
|
"content": content,
|
|
}
|
|
|
|
|
|
def test_valid_arguments_unchanged():
|
|
messages = [
|
|
{"role": "user", "content": "hello"},
|
|
_assistant_message(_tool_call(arguments='{"path":"/tmp/foo"}')),
|
|
_tool_message(content="done"),
|
|
]
|
|
original = copy.deepcopy(messages)
|
|
|
|
repaired = AIAgent._sanitize_tool_call_arguments(messages)
|
|
|
|
assert repaired == 0
|
|
assert messages == original
|
|
|
|
|
|
def test_truncated_arguments_replaced_with_empty_object(caplog):
|
|
messages = [
|
|
_assistant_message(_tool_call(arguments='{"path": "/tmp/foo')),
|
|
]
|
|
|
|
with caplog.at_level(logging.WARNING, logger="run_agent"):
|
|
repaired = AIAgent._sanitize_tool_call_arguments(
|
|
messages,
|
|
logger=logging.getLogger("run_agent"),
|
|
session_id="session-123",
|
|
)
|
|
|
|
assert repaired == 1
|
|
assert messages[0]["tool_calls"][0]["function"]["arguments"] == "{}"
|
|
assert any(
|
|
"session=session-123" in record.message
|
|
and "tool_call_id=call_1" in record.message
|
|
for record in caplog.records
|
|
)
|
|
|
|
|
|
def test_marker_appended_to_existing_tool_message():
|
|
marker = AIAgent._TOOL_CALL_ARGUMENTS_CORRUPTION_MARKER
|
|
messages = [
|
|
_assistant_message(_tool_call(arguments='{"path": "/tmp/foo')),
|
|
_tool_message(content="existing tool output"),
|
|
]
|
|
|
|
repaired = AIAgent._sanitize_tool_call_arguments(messages)
|
|
|
|
assert repaired == 1
|
|
assert messages[1]["content"] == f"{marker}\nexisting tool output"
|
|
|
|
|
|
def test_marker_message_inserted_when_missing():
|
|
marker = AIAgent._TOOL_CALL_ARGUMENTS_CORRUPTION_MARKER
|
|
messages = [
|
|
_assistant_message(_tool_call(arguments='{"path": "/tmp/foo')),
|
|
{"role": "user", "content": "next turn"},
|
|
]
|
|
|
|
repaired = AIAgent._sanitize_tool_call_arguments(messages)
|
|
|
|
assert repaired == 1
|
|
assert messages[1] == {
|
|
"role": "tool",
|
|
"name": "read_file",
|
|
"tool_call_id": "call_1",
|
|
"content": marker,
|
|
}
|
|
assert messages[2] == {"role": "user", "content": "next turn"}
|
|
|
|
|
|
def test_multiple_corrupted_tool_calls_in_one_message():
|
|
marker = AIAgent._TOOL_CALL_ARGUMENTS_CORRUPTION_MARKER
|
|
messages = [
|
|
_assistant_message(
|
|
_tool_call(call_id="call_1", arguments='{"path": "/tmp/foo'),
|
|
_tool_call(call_id="call_2", arguments='{"path":"/tmp/bar"}'),
|
|
_tool_call(call_id="call_3", arguments='{"mode":"tail"'),
|
|
),
|
|
]
|
|
|
|
repaired = AIAgent._sanitize_tool_call_arguments(messages)
|
|
|
|
assert repaired == 2
|
|
assert messages[0]["tool_calls"][0]["function"]["arguments"] == "{}"
|
|
assert messages[0]["tool_calls"][1]["function"]["arguments"] == '{"path":"/tmp/bar"}'
|
|
assert messages[0]["tool_calls"][2]["function"]["arguments"] == "{}"
|
|
assert messages[1]["tool_call_id"] == "call_1"
|
|
assert messages[1]["content"] == marker
|
|
assert messages[2]["tool_call_id"] == "call_3"
|
|
assert messages[2]["content"] == marker
|
|
|
|
|
|
def test_empty_string_arguments_treated_as_empty_object(caplog):
|
|
messages = [
|
|
_assistant_message(_tool_call(arguments="")),
|
|
]
|
|
|
|
with caplog.at_level(logging.WARNING, logger="run_agent"):
|
|
repaired = AIAgent._sanitize_tool_call_arguments(
|
|
messages,
|
|
logger=logging.getLogger("run_agent"),
|
|
session_id="session-123",
|
|
)
|
|
|
|
assert repaired == 0
|
|
assert messages[0]["tool_calls"][0]["function"]["arguments"] == "{}"
|
|
assert caplog.records == []
|
|
|
|
|
|
def test_non_assistant_messages_ignored():
|
|
messages = [
|
|
{"role": "user", "content": "hello", "tool_calls": [_tool_call(arguments='{"bad":')]},
|
|
{"role": "tool", "tool_call_id": "call_1", "content": "ok"},
|
|
{"role": "system", "content": "sys", "tool_calls": [_tool_call(arguments='{"bad":')]},
|
|
None,
|
|
"not a dict",
|
|
]
|
|
original = copy.deepcopy(messages)
|
|
|
|
repaired = AIAgent._sanitize_tool_call_arguments(messages)
|
|
|
|
assert repaired == 0
|
|
assert messages == original
|