Skip to content
34 changes: 20 additions & 14 deletions src/crewai/tools/tool_usage.py
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@ def __init__(
agent: Optional[Union["BaseAgent", "LiteAgent"]] = None,
action: Any = None,
fingerprint_context: Optional[Dict[str, str]] = None,
original_tools: List[Any] = [],
) -> None:
self._i18n: I18N = agent.i18n if agent else I18N()
self._printer: Printer = Printer()
Expand All @@ -86,6 +87,7 @@ def __init__(
self.tools_description = render_text_description_and_args(tools)
self.tools_names = get_tool_names(tools)
self.tools_handler = tools_handler
self.original_tools = original_tools
self.tools = tools
self.task = task
self.action = action
Expand Down Expand Up @@ -191,13 +193,16 @@ def _use(
) # type: ignore
from_cache = result is not None

original_tool = None
if hasattr(self, 'original_tools') and self.original_tools:
original_tool = next(
(ot for ot in self.original_tools if ot.name == tool.name),
None
)

available_tool = next(
(
available_tool
for available_tool in self.tools
if available_tool.name == tool.name
),
None,
(at for at in self.tools if at.name == tool.name),
None
)

if result is None:
Expand Down Expand Up @@ -259,10 +264,11 @@ def _use(

if self.tools_handler:
should_cache = True
if (
hasattr(available_tool, "cache_function")
and available_tool.cache_function # type: ignore # Item "None" of "Any | None" has no attribute "cache_function"
):
if original_tool and hasattr(original_tool, "cache_function") and original_tool.cache_function:
should_cache = original_tool.cache_function(
calling.arguments, result
)
elif available_tool and hasattr(available_tool, "cache_function") and available_tool.cache_function:
should_cache = available_tool.cache_function( # type: ignore # Item "None" of "Any | None" has no attribute "cache_function"
calling.arguments, result
)
Expand Down Expand Up @@ -290,10 +296,10 @@ def _use(
result=result,
)

if (
hasattr(available_tool, "result_as_answer")
and available_tool.result_as_answer # type: ignore # Item "None" of "Any | None" has no attribute "cache_function"
):
if original_tool and hasattr(original_tool, "result_as_answer") and original_tool.result_as_answer:
result_as_answer = original_tool.result_as_answer
data["result_as_answer"] = result_as_answer
elif available_tool and hasattr(available_tool, "result_as_answer") and available_tool.result_as_answer:
result_as_answer = available_tool.result_as_answer # type: ignore # Item "None" of "Any | None" has no attribute "result_as_answer"
data["result_as_answer"] = result_as_answer # type: ignore

Expand Down
1 change: 1 addition & 0 deletions src/crewai/utilities/tool_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,7 @@ def execute_tool_and_check_finality(
task=task,
agent=agent,
action=agent_action,
original_tools=tools, # Pass original tools to ensure custom tools work
)

# Parse tool calling
Expand Down
77 changes: 77 additions & 0 deletions tests/tools/test_custom_tool_invocation.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
from unittest.mock import MagicMock

import pytest
from pydantic import BaseModel, Field

from crewai import Agent, Task
from crewai.agents.crew_agent_executor import CrewAgentExecutor
from crewai.agents.parser import AgentAction
from crewai.agents.tools_handler import ToolsHandler
from crewai.tools import BaseTool
from crewai.utilities.i18n import I18N
from crewai.utilities.tool_utils import execute_tool_and_check_finality


class TestToolInput(BaseModel):
test_param: str = Field(..., description="A test parameter")


class TestCustomTool(BaseTool):
name: str = "Test Custom Tool"
description: str = "A test tool to verify custom tool invocation"
args_schema: type[BaseModel] = TestToolInput

def _run(self, test_param: str) -> str:
return f"Tool executed with param: {test_param}"


def test_custom_tool_invocation():
custom_tool = TestCustomTool()

mock_agent = MagicMock()
mock_task = MagicMock()
mock_llm = MagicMock()
mock_crew = MagicMock()
tools_handler = ToolsHandler()

executor = CrewAgentExecutor(
llm=mock_llm,
task=mock_task,
crew=mock_crew,
agent=mock_agent,
prompt={},
max_iter=5,
tools=[custom_tool],
tools_names="Test Custom Tool",
stop_words=[],
tools_description="A test tool to verify custom tool invocation",
tools_handler=tools_handler,
original_tools=[custom_tool]
)

action = AgentAction(
tool="Test Custom Tool",
tool_input={"test_param": "test_value"},
thought="I'll use the custom tool",
text="I'll use the Test Custom Tool to get a result"
)

i18n = I18N()

mock_agent.key = "test_agent"
mock_agent.role = "test_role"

result = execute_tool_and_check_finality(
agent_action=action,
tools=[custom_tool],
i18n=i18n,
agent_key=mock_agent.key,
agent_role=mock_agent.role,
tools_handler=tools_handler,
task=mock_task,
agent=mock_agent,
function_calling_llm=mock_llm
)

assert "Tool executed with param: test_value" in result.result
assert result.result_as_answer is False
Loading