mirror of
https://github.com/hwchase17/langchain.git
synced 2025-08-31 02:11:09 +00:00
Changes: - ToolCall, InvalidToolCall and ToolCallChunk can all accept a "type" parameter now - LLM integration packages add "type" to all the above - Tool supports ToolCall inputs that have "type" specified - Tool outputs ToolMessage when a ToolCall is passed as input - Tools can separately specify ToolMessage.content and ToolMessage.raw_output - Tools emit events for validation errors (using on_tool_error and on_tool_end) Example: ```python @tool("structured_api", response_format="content_and_raw_output") def _mock_structured_tool_with_raw_output( arg1: int, arg2: bool, arg3: Optional[dict] = None ) -> Tuple[str, dict]: """A Structured Tool""" return f"{arg1} {arg2}", {"arg1": arg1, "arg2": arg2, "arg3": arg3} def test_tool_call_input_tool_message_with_raw_output() -> None: tool_call: Dict = { "name": "structured_api", "args": {"arg1": 1, "arg2": True, "arg3": {"img": "base64string..."}}, "id": "123", "type": "tool_call", } expected = ToolMessage("1 True", raw_output=tool_call["args"], tool_call_id="123") tool = _mock_structured_tool_with_raw_output actual = tool.invoke(tool_call) assert actual == expected tool_call.pop("type") with pytest.raises(ValidationError): tool.invoke(tool_call) actual_content = tool.invoke(tool_call["args"]) assert actual_content == expected.content ``` --------- Co-authored-by: Erick Friis <erick@langchain.dev>
114 lines
2.9 KiB
Python
114 lines
2.9 KiB
Python
from typing import Any, List, Literal
|
|
|
|
from langchain_core.messages import AIMessage
|
|
from langchain_core.outputs import ChatGeneration
|
|
from langchain_core.pydantic_v1 import BaseModel
|
|
|
|
from langchain_anthropic.output_parsers import ToolsOutputParser
|
|
|
|
_CONTENT: List = [
|
|
{
|
|
"type": "text",
|
|
"text": "thought",
|
|
},
|
|
{"type": "tool_use", "input": {"bar": 0}, "id": "1", "name": "_Foo1"},
|
|
{
|
|
"type": "text",
|
|
"text": "thought",
|
|
},
|
|
{"type": "tool_use", "input": {"baz": "a"}, "id": "2", "name": "_Foo2"},
|
|
]
|
|
|
|
_RESULT: List = [ChatGeneration(message=AIMessage(_CONTENT))] # type: ignore[misc]
|
|
|
|
|
|
class _Foo1(BaseModel):
|
|
bar: int
|
|
|
|
|
|
class _Foo2(BaseModel):
|
|
baz: Literal["a", "b"]
|
|
|
|
|
|
def test_tools_output_parser() -> None:
|
|
output_parser = ToolsOutputParser()
|
|
expected = [
|
|
{
|
|
"name": "_Foo1",
|
|
"args": {"bar": 0},
|
|
"id": "1",
|
|
"index": 1,
|
|
"type": "tool_call",
|
|
},
|
|
{
|
|
"name": "_Foo2",
|
|
"args": {"baz": "a"},
|
|
"id": "2",
|
|
"index": 3,
|
|
"type": "tool_call",
|
|
},
|
|
]
|
|
actual = output_parser.parse_result(_RESULT)
|
|
assert expected == actual
|
|
|
|
|
|
def test_tools_output_parser_args_only() -> None:
|
|
output_parser = ToolsOutputParser(args_only=True)
|
|
expected = [
|
|
{"bar": 0},
|
|
{"baz": "a"},
|
|
]
|
|
actual = output_parser.parse_result(_RESULT)
|
|
assert expected == actual
|
|
|
|
expected = []
|
|
actual = output_parser.parse_result([ChatGeneration(message=AIMessage(""))]) # type: ignore[misc]
|
|
assert expected == actual
|
|
|
|
|
|
def test_tools_output_parser_first_tool_only() -> None:
|
|
output_parser = ToolsOutputParser(first_tool_only=True)
|
|
expected: Any = {
|
|
"name": "_Foo1",
|
|
"args": {"bar": 0},
|
|
"id": "1",
|
|
"index": 1,
|
|
"type": "tool_call",
|
|
}
|
|
actual = output_parser.parse_result(_RESULT)
|
|
assert expected == actual
|
|
|
|
expected = None
|
|
actual = output_parser.parse_result([ChatGeneration(message=AIMessage(""))]) # type: ignore[misc]
|
|
assert expected == actual
|
|
|
|
|
|
def test_tools_output_parser_pydantic() -> None:
|
|
output_parser = ToolsOutputParser(pydantic_schemas=[_Foo1, _Foo2])
|
|
expected = [_Foo1(bar=0), _Foo2(baz="a")]
|
|
actual = output_parser.parse_result(_RESULT)
|
|
assert expected == actual
|
|
|
|
|
|
def test_tools_output_parser_empty_content() -> None:
|
|
class ChartType(BaseModel):
|
|
chart_type: Literal["pie", "line", "bar"]
|
|
|
|
output_parser = ToolsOutputParser(
|
|
first_tool_only=True, pydantic_schemas=[ChartType]
|
|
)
|
|
message = AIMessage(
|
|
"",
|
|
tool_calls=[
|
|
{
|
|
"name": "ChartType",
|
|
"args": {"chart_type": "pie"},
|
|
"id": "foo",
|
|
"type": "tool_call",
|
|
}
|
|
],
|
|
)
|
|
actual = output_parser.invoke(message)
|
|
expected = ChartType(chart_type="pie")
|
|
assert expected == actual
|