You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

api_entities.py 3.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. from datetime import datetime
  2. from typing import Any, Literal, Optional
  3. from pydantic import BaseModel, Field, field_validator
  4. from core.model_runtime.utils.encoders import jsonable_encoder
  5. from core.tools.__base.tool import ToolParameter
  6. from core.tools.entities.common_entities import I18nObject
  7. from core.tools.entities.tool_entities import ToolProviderType
  8. class ToolApiEntity(BaseModel):
  9. author: str
  10. name: str # identifier
  11. label: I18nObject # label
  12. description: I18nObject
  13. parameters: Optional[list[ToolParameter]] = None
  14. labels: list[str] = Field(default_factory=list)
  15. output_schema: Optional[dict] = None
  16. ToolProviderTypeApiLiteral = Optional[Literal["builtin", "api", "workflow", "mcp"]]
  17. class ToolProviderApiEntity(BaseModel):
  18. id: str
  19. author: str
  20. name: str # identifier
  21. description: I18nObject
  22. icon: str | dict
  23. label: I18nObject # label
  24. type: ToolProviderType
  25. masked_credentials: Optional[dict] = None
  26. original_credentials: Optional[dict] = None
  27. is_team_authorization: bool = False
  28. allow_delete: bool = True
  29. plugin_id: Optional[str] = Field(default="", description="The plugin id of the tool")
  30. plugin_unique_identifier: Optional[str] = Field(default="", description="The unique identifier of the tool")
  31. tools: list[ToolApiEntity] = Field(default_factory=list)
  32. labels: list[str] = Field(default_factory=list)
  33. # MCP
  34. server_url: Optional[str] = Field(default="", description="The server url of the tool")
  35. updated_at: int = Field(default_factory=lambda: int(datetime.now().timestamp()))
  36. server_identifier: Optional[str] = Field(default="", description="The server identifier of the MCP tool")
  37. @field_validator("tools", mode="before")
  38. @classmethod
  39. def convert_none_to_empty_list(cls, v):
  40. return v if v is not None else []
  41. def to_dict(self) -> dict:
  42. # -------------
  43. # overwrite tool parameter types for temp fix
  44. tools = jsonable_encoder(self.tools)
  45. for tool in tools:
  46. if tool.get("parameters"):
  47. for parameter in tool.get("parameters"):
  48. if parameter.get("type") == ToolParameter.ToolParameterType.SYSTEM_FILES.value:
  49. parameter["type"] = "files"
  50. if parameter.get("input_schema") is None:
  51. parameter.pop("input_schema", None)
  52. # -------------
  53. optional_fields = self.optional_field("server_url", self.server_url)
  54. if self.type == ToolProviderType.MCP.value:
  55. optional_fields.update(self.optional_field("updated_at", self.updated_at))
  56. optional_fields.update(self.optional_field("server_identifier", self.server_identifier))
  57. return {
  58. "id": self.id,
  59. "author": self.author,
  60. "name": self.name,
  61. "plugin_id": self.plugin_id,
  62. "plugin_unique_identifier": self.plugin_unique_identifier,
  63. "description": self.description.to_dict(),
  64. "icon": self.icon,
  65. "label": self.label.to_dict(),
  66. "type": self.type.value,
  67. "team_credentials": self.masked_credentials,
  68. "is_team_authorization": self.is_team_authorization,
  69. "allow_delete": self.allow_delete,
  70. "tools": tools,
  71. "labels": self.labels,
  72. **optional_fields,
  73. }
  74. def optional_field(self, key: str, value: Any) -> dict:
  75. """Return dict with key-value if value is truthy, empty dict otherwise."""
  76. return {key: value} if value else {}