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.

plugin.py 7.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. import datetime
  2. import enum
  3. import re
  4. from collections.abc import Mapping
  5. from typing import Any, Optional
  6. from pydantic import BaseModel, Field, model_validator
  7. from werkzeug.exceptions import NotFound
  8. from core.agent.plugin_entities import AgentStrategyProviderEntity
  9. from core.model_runtime.entities.provider_entities import ProviderEntity
  10. from core.plugin.entities.base import BasePluginEntity
  11. from core.plugin.entities.endpoint import EndpointProviderDeclaration
  12. from core.tools.entities.common_entities import I18nObject
  13. from core.tools.entities.tool_entities import ToolProviderEntity
  14. class PluginInstallationSource(enum.StrEnum):
  15. Github = "github"
  16. Marketplace = "marketplace"
  17. Package = "package"
  18. Remote = "remote"
  19. class PluginResourceRequirements(BaseModel):
  20. memory: int
  21. class Permission(BaseModel):
  22. class Tool(BaseModel):
  23. enabled: Optional[bool] = Field(default=False)
  24. class Model(BaseModel):
  25. enabled: Optional[bool] = Field(default=False)
  26. llm: Optional[bool] = Field(default=False)
  27. text_embedding: Optional[bool] = Field(default=False)
  28. rerank: Optional[bool] = Field(default=False)
  29. tts: Optional[bool] = Field(default=False)
  30. speech2text: Optional[bool] = Field(default=False)
  31. moderation: Optional[bool] = Field(default=False)
  32. class Node(BaseModel):
  33. enabled: Optional[bool] = Field(default=False)
  34. class Endpoint(BaseModel):
  35. enabled: Optional[bool] = Field(default=False)
  36. class Storage(BaseModel):
  37. enabled: Optional[bool] = Field(default=False)
  38. size: int = Field(ge=1024, le=1073741824, default=1048576)
  39. tool: Optional[Tool] = Field(default=None)
  40. model: Optional[Model] = Field(default=None)
  41. node: Optional[Node] = Field(default=None)
  42. endpoint: Optional[Endpoint] = Field(default=None)
  43. storage: Optional[Storage] = Field(default=None)
  44. permission: Optional[Permission] = Field(default=None)
  45. class PluginCategory(enum.StrEnum):
  46. Tool = "tool"
  47. Model = "model"
  48. Extension = "extension"
  49. AgentStrategy = "agent-strategy"
  50. class PluginDeclaration(BaseModel):
  51. class Plugins(BaseModel):
  52. tools: Optional[list[str]] = Field(default_factory=list[str])
  53. models: Optional[list[str]] = Field(default_factory=list[str])
  54. endpoints: Optional[list[str]] = Field(default_factory=list[str])
  55. class Meta(BaseModel):
  56. minimum_dify_version: Optional[str] = Field(default=None, pattern=r"^\d{1,4}(\.\d{1,4}){1,3}(-\w{1,16})?$")
  57. version: Optional[str] = Field(default=None)
  58. version: str = Field(..., pattern=r"^\d{1,4}(\.\d{1,4}){1,3}(-\w{1,16})?$")
  59. author: Optional[str] = Field(..., pattern=r"^[a-zA-Z0-9_-]{1,64}$")
  60. name: str = Field(..., pattern=r"^[a-z0-9_-]{1,128}$")
  61. description: I18nObject
  62. icon: str
  63. icon_dark: Optional[str] = Field(default=None)
  64. label: I18nObject
  65. category: PluginCategory
  66. created_at: datetime.datetime
  67. resource: PluginResourceRequirements
  68. plugins: Plugins
  69. tags: list[str] = Field(default_factory=list)
  70. repo: Optional[str] = Field(default=None)
  71. verified: bool = Field(default=False)
  72. tool: Optional[ToolProviderEntity] = None
  73. model: Optional[ProviderEntity] = None
  74. endpoint: Optional[EndpointProviderDeclaration] = None
  75. agent_strategy: Optional[AgentStrategyProviderEntity] = None
  76. meta: Meta
  77. @model_validator(mode="before")
  78. @classmethod
  79. def validate_category(cls, values: dict) -> dict:
  80. # auto detect category
  81. if values.get("tool"):
  82. values["category"] = PluginCategory.Tool
  83. elif values.get("model"):
  84. values["category"] = PluginCategory.Model
  85. elif values.get("agent_strategy"):
  86. values["category"] = PluginCategory.AgentStrategy
  87. else:
  88. values["category"] = PluginCategory.Extension
  89. return values
  90. class PluginInstallation(BasePluginEntity):
  91. tenant_id: str
  92. endpoints_setups: int
  93. endpoints_active: int
  94. runtime_type: str
  95. source: PluginInstallationSource
  96. meta: Mapping[str, Any]
  97. plugin_id: str
  98. plugin_unique_identifier: str
  99. version: str
  100. checksum: str
  101. declaration: PluginDeclaration
  102. class PluginEntity(PluginInstallation):
  103. name: str
  104. installation_id: str
  105. version: str
  106. @model_validator(mode="after")
  107. def set_plugin_id(self):
  108. if self.declaration.tool:
  109. self.declaration.tool.plugin_id = self.plugin_id
  110. return self
  111. class GithubPackage(BaseModel):
  112. repo: str
  113. version: str
  114. package: str
  115. class GithubVersion(BaseModel):
  116. repo: str
  117. version: str
  118. class GenericProviderID:
  119. organization: str
  120. plugin_name: str
  121. provider_name: str
  122. is_hardcoded: bool
  123. def to_string(self) -> str:
  124. return str(self)
  125. def __str__(self) -> str:
  126. return f"{self.organization}/{self.plugin_name}/{self.provider_name}"
  127. def __init__(self, value: str, is_hardcoded: bool = False) -> None:
  128. if not value:
  129. raise NotFound("plugin not found, please add plugin")
  130. # check if the value is a valid plugin id with format: $organization/$plugin_name/$provider_name
  131. if not re.match(r"^[a-z0-9_-]+\/[a-z0-9_-]+\/[a-z0-9_-]+$", value):
  132. # check if matches [a-z0-9_-]+, if yes, append with langgenius/$value/$value
  133. if re.match(r"^[a-z0-9_-]+$", value):
  134. value = f"langgenius/{value}/{value}"
  135. else:
  136. raise ValueError(f"Invalid plugin id {value}")
  137. self.organization, self.plugin_name, self.provider_name = value.split("/")
  138. self.is_hardcoded = is_hardcoded
  139. def is_langgenius(self) -> bool:
  140. return self.organization == "langgenius"
  141. @property
  142. def plugin_id(self) -> str:
  143. return f"{self.organization}/{self.plugin_name}"
  144. class ModelProviderID(GenericProviderID):
  145. def __init__(self, value: str, is_hardcoded: bool = False) -> None:
  146. super().__init__(value, is_hardcoded)
  147. if self.organization == "langgenius" and self.provider_name == "google":
  148. self.plugin_name = "gemini"
  149. class ToolProviderID(GenericProviderID):
  150. def __init__(self, value: str, is_hardcoded: bool = False) -> None:
  151. super().__init__(value, is_hardcoded)
  152. if self.organization == "langgenius":
  153. if self.provider_name in ["jina", "siliconflow", "stepfun", "gitee_ai"]:
  154. self.plugin_name = f"{self.provider_name}_tool"
  155. class PluginDependency(BaseModel):
  156. class Type(enum.StrEnum):
  157. Github = PluginInstallationSource.Github.value
  158. Marketplace = PluginInstallationSource.Marketplace.value
  159. Package = PluginInstallationSource.Package.value
  160. class Github(BaseModel):
  161. repo: str
  162. version: str
  163. package: str
  164. github_plugin_unique_identifier: str
  165. @property
  166. def plugin_unique_identifier(self) -> str:
  167. return self.github_plugin_unique_identifier
  168. class Marketplace(BaseModel):
  169. marketplace_plugin_unique_identifier: str
  170. @property
  171. def plugin_unique_identifier(self) -> str:
  172. return self.marketplace_plugin_unique_identifier
  173. class Package(BaseModel):
  174. plugin_unique_identifier: str
  175. type: Type
  176. value: Github | Marketplace | Package
  177. current_identifier: Optional[str] = None
  178. class MissingPluginDependency(BaseModel):
  179. plugin_unique_identifier: str
  180. current_identifier: Optional[str] = None