mirror of
https://github.com/langgenius/dify.git
synced 2025-11-12 09:25:17 +00:00
Signed-off-by: -LAN- <laipz8200@outlook.com> Co-authored-by: twwu <twwu@dify.ai> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: jyong <718720800@qq.com> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: QuantumGhost <obelisk.reg+git@gmail.com> Co-authored-by: lyzno1 <yuanyouhuilyz@gmail.com> Co-authored-by: quicksand <quicksandzn@gmail.com> Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com> Co-authored-by: lyzno1 <92089059+lyzno1@users.noreply.github.com> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: Yongtao Huang <yongtaoh2022@gmail.com> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: nite-knite <nkCoding@gmail.com> Co-authored-by: Hanqing Zhao <sherry9277@gmail.com> Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com> Co-authored-by: Harry <xh001x@hotmail.com>
118 lines
5.5 KiB
Python
118 lines
5.5 KiB
Python
from collections.abc import Mapping
|
|
from datetime import datetime
|
|
from typing import Any, Literal
|
|
|
|
from pydantic import BaseModel, Field, field_validator
|
|
|
|
from core.model_runtime.utils.encoders import jsonable_encoder
|
|
from core.tools.__base.tool import ToolParameter
|
|
from core.tools.entities.common_entities import I18nObject
|
|
from core.tools.entities.tool_entities import CredentialType, ToolProviderType
|
|
|
|
|
|
class ToolApiEntity(BaseModel):
|
|
author: str
|
|
name: str # identifier
|
|
label: I18nObject # label
|
|
description: I18nObject
|
|
parameters: list[ToolParameter] | None = None
|
|
labels: list[str] = Field(default_factory=list)
|
|
output_schema: Mapping[str, object] = Field(default_factory=dict)
|
|
|
|
|
|
ToolProviderTypeApiLiteral = Literal["builtin", "api", "workflow", "mcp"] | None
|
|
|
|
|
|
class ToolProviderApiEntity(BaseModel):
|
|
id: str
|
|
author: str
|
|
name: str # identifier
|
|
description: I18nObject
|
|
icon: str | Mapping[str, str]
|
|
icon_dark: str | Mapping[str, str] = ""
|
|
label: I18nObject # label
|
|
type: ToolProviderType
|
|
masked_credentials: Mapping[str, object] = Field(default_factory=dict)
|
|
original_credentials: Mapping[str, object] = Field(default_factory=dict)
|
|
is_team_authorization: bool = False
|
|
allow_delete: bool = True
|
|
plugin_id: str | None = Field(default="", description="The plugin id of the tool")
|
|
plugin_unique_identifier: str | None = Field(default="", description="The unique identifier of the tool")
|
|
tools: list[ToolApiEntity] = Field(default_factory=list[ToolApiEntity])
|
|
labels: list[str] = Field(default_factory=list)
|
|
# MCP
|
|
server_url: str | None = Field(default="", description="The server url of the tool")
|
|
updated_at: int = Field(default_factory=lambda: int(datetime.now().timestamp()))
|
|
server_identifier: str | None = Field(default="", description="The server identifier of the MCP tool")
|
|
timeout: float | None = Field(default=30.0, description="The timeout of the MCP tool")
|
|
sse_read_timeout: float | None = Field(default=300.0, description="The SSE read timeout of the MCP tool")
|
|
masked_headers: dict[str, str] | None = Field(default=None, description="The masked headers of the MCP tool")
|
|
original_headers: dict[str, str] | None = Field(default=None, description="The original headers of the MCP tool")
|
|
|
|
@field_validator("tools", mode="before")
|
|
@classmethod
|
|
def convert_none_to_empty_list(cls, v):
|
|
return v if v is not None else []
|
|
|
|
def to_dict(self):
|
|
# -------------
|
|
# overwrite tool parameter types for temp fix
|
|
tools = jsonable_encoder(self.tools)
|
|
for tool in tools:
|
|
if tool.get("parameters"):
|
|
for parameter in tool.get("parameters"):
|
|
if parameter.get("type") == ToolParameter.ToolParameterType.SYSTEM_FILES.value:
|
|
parameter["type"] = "files"
|
|
if parameter.get("input_schema") is None:
|
|
parameter.pop("input_schema", None)
|
|
# -------------
|
|
optional_fields = self.optional_field("server_url", self.server_url)
|
|
if self.type == ToolProviderType.MCP:
|
|
optional_fields.update(self.optional_field("updated_at", self.updated_at))
|
|
optional_fields.update(self.optional_field("server_identifier", self.server_identifier))
|
|
optional_fields.update(self.optional_field("timeout", self.timeout))
|
|
optional_fields.update(self.optional_field("sse_read_timeout", self.sse_read_timeout))
|
|
optional_fields.update(self.optional_field("masked_headers", self.masked_headers))
|
|
optional_fields.update(self.optional_field("original_headers", self.original_headers))
|
|
return {
|
|
"id": self.id,
|
|
"author": self.author,
|
|
"name": self.name,
|
|
"plugin_id": self.plugin_id,
|
|
"plugin_unique_identifier": self.plugin_unique_identifier,
|
|
"description": self.description.to_dict(),
|
|
"icon": self.icon,
|
|
"icon_dark": self.icon_dark,
|
|
"label": self.label.to_dict(),
|
|
"type": self.type.value,
|
|
"team_credentials": self.masked_credentials,
|
|
"is_team_authorization": self.is_team_authorization,
|
|
"allow_delete": self.allow_delete,
|
|
"tools": tools,
|
|
"labels": self.labels,
|
|
**optional_fields,
|
|
}
|
|
|
|
def optional_field(self, key: str, value: Any):
|
|
"""Return dict with key-value if value is truthy, empty dict otherwise."""
|
|
return {key: value} if value else {}
|
|
|
|
|
|
class ToolProviderCredentialApiEntity(BaseModel):
|
|
id: str = Field(description="The unique id of the credential")
|
|
name: str = Field(description="The name of the credential")
|
|
provider: str = Field(description="The provider of the credential")
|
|
credential_type: CredentialType = Field(description="The type of the credential")
|
|
is_default: bool = Field(
|
|
default=False, description="Whether the credential is the default credential for the provider in the workspace"
|
|
)
|
|
credentials: Mapping[str, object] = Field(description="The credentials of the provider", default_factory=dict)
|
|
|
|
|
|
class ToolProviderCredentialInfoApiEntity(BaseModel):
|
|
supported_credential_types: list[str] = Field(description="The supported credential types of the provider")
|
|
is_oauth_custom_client_enabled: bool = Field(
|
|
default=False, description="Whether the OAuth custom client is enabled for the provider"
|
|
)
|
|
credentials: list[ToolProviderCredentialApiEntity] = Field(description="The credentials of the provider")
|