dify/api/core/plugin/backwards_invocation/tool.py

85 lines
3.1 KiB
Python

from collections.abc import Generator
from typing import Any
from core.callback_handler.workflow_tool_callback_handler import DifyWorkflowCallbackHandler
from core.plugin.backwards_invocation.base import BaseBackwardsInvocation
from core.tools.entities.tool_entities import ToolInvokeMessage, ToolProviderType
from core.tools.signature import sign_tool_file
from core.tools.tool_engine import ToolEngine
from core.tools.tool_manager import ToolManager
from core.tools.utils.message_transformer import ToolFileMessageTransformer
class PluginToolBackwardsInvocation(BaseBackwardsInvocation):
"""
Backwards invocation for plugin tools.
"""
@classmethod
def invoke_tool(
cls,
tenant_id: str,
user_id: str,
tool_type: ToolProviderType,
provider: str,
tool_name: str,
tool_parameters: dict[str, Any],
credential_id: str | None = None,
) -> Generator[ToolInvokeMessage, None, None]:
"""
invoke tool
"""
# get tool runtime
try:
tool_runtime = ToolManager.get_tool_runtime_from_plugin(
tool_type, tenant_id, provider, tool_name, tool_parameters, credential_id
)
response = ToolEngine.generic_invoke(
tool_runtime, tool_parameters, user_id, DifyWorkflowCallbackHandler(), workflow_call_depth=1
)
response = ToolFileMessageTransformer.transform_tool_invoke_messages(
response, user_id=user_id, tenant_id=tenant_id
)
return cls._sign_tool_file_urls(response)
except Exception as e:
raise e
# FIXME: this method should be gracefully deprecated
@classmethod
def _sign_tool_file_urls(
cls, messages: Generator[ToolInvokeMessage, None, None]
) -> Generator[ToolInvokeMessage, None, None]:
"""
Sign file URLs in tool invoke messages for external access.
"""
for message in messages:
if message.type in {
ToolInvokeMessage.MessageType.IMAGE_LINK,
ToolInvokeMessage.MessageType.BINARY_LINK,
ToolInvokeMessage.MessageType.FILE,
}:
if isinstance(message.message, ToolInvokeMessage.TextMessage):
url = message.message.text
# Check if it's an unsigned internal path
if url.startswith("/files/tools/"):
parts = url.split("/")[-1]
if "." in parts:
file_id, ext = parts.rsplit(".", 1)
extension = f".{ext}"
else:
file_id = parts
extension = ".bin"
signed_url = sign_tool_file(tool_file_id=file_id, extension=extension)
yield ToolInvokeMessage(
type=message.type,
message=ToolInvokeMessage.TextMessage(text=signed_url),
meta=message.meta,
)
continue
yield message