Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. from collections.abc import Generator
  2. from typing import Any, Optional
  3. from pydantic import BaseModel
  4. from core.plugin.entities.plugin import GenericProviderID, ToolProviderID
  5. from core.plugin.entities.plugin_daemon import PluginBasicBooleanResponse, PluginToolProviderEntity
  6. from core.plugin.impl.base import BasePluginClient
  7. from core.plugin.utils.chunk_merger import merge_blob_chunks
  8. from core.tools.entities.tool_entities import CredentialType, ToolInvokeMessage, ToolParameter
  9. class PluginToolManager(BasePluginClient):
  10. def fetch_tool_providers(self, tenant_id: str) -> list[PluginToolProviderEntity]:
  11. """
  12. Fetch tool providers for the given tenant.
  13. """
  14. def transformer(json_response: dict[str, Any]):
  15. for provider in json_response.get("data", []):
  16. declaration = provider.get("declaration", {}) or {}
  17. provider_name = declaration.get("identity", {}).get("name")
  18. for tool in declaration.get("tools", []):
  19. tool["identity"]["provider"] = provider_name
  20. return json_response
  21. response = self._request_with_plugin_daemon_response(
  22. "GET",
  23. f"plugin/{tenant_id}/management/tools",
  24. list[PluginToolProviderEntity],
  25. params={"page": 1, "page_size": 256},
  26. transformer=transformer,
  27. )
  28. for provider in response:
  29. provider.declaration.identity.name = f"{provider.plugin_id}/{provider.declaration.identity.name}"
  30. # override the provider name for each tool to plugin_id/provider_name
  31. for tool in provider.declaration.tools:
  32. tool.identity.provider = provider.declaration.identity.name
  33. return response
  34. def fetch_tool_provider(self, tenant_id: str, provider: str) -> PluginToolProviderEntity:
  35. """
  36. Fetch tool provider for the given tenant and plugin.
  37. """
  38. tool_provider_id = ToolProviderID(provider)
  39. def transformer(json_response: dict[str, Any]):
  40. data = json_response.get("data")
  41. if data:
  42. for tool in data.get("declaration", {}).get("tools", []):
  43. tool["identity"]["provider"] = tool_provider_id.provider_name
  44. return json_response
  45. response = self._request_with_plugin_daemon_response(
  46. "GET",
  47. f"plugin/{tenant_id}/management/tool",
  48. PluginToolProviderEntity,
  49. params={"provider": tool_provider_id.provider_name, "plugin_id": tool_provider_id.plugin_id},
  50. transformer=transformer,
  51. )
  52. response.declaration.identity.name = f"{response.plugin_id}/{response.declaration.identity.name}"
  53. # override the provider name for each tool to plugin_id/provider_name
  54. for tool in response.declaration.tools:
  55. tool.identity.provider = response.declaration.identity.name
  56. return response
  57. def invoke(
  58. self,
  59. tenant_id: str,
  60. user_id: str,
  61. tool_provider: str,
  62. tool_name: str,
  63. credentials: dict[str, Any],
  64. credential_type: CredentialType,
  65. tool_parameters: dict[str, Any],
  66. conversation_id: Optional[str] = None,
  67. app_id: Optional[str] = None,
  68. message_id: Optional[str] = None,
  69. ) -> Generator[ToolInvokeMessage, None, None]:
  70. """
  71. Invoke the tool with the given tenant, user, plugin, provider, name, credentials and parameters.
  72. """
  73. tool_provider_id = GenericProviderID(tool_provider)
  74. response = self._request_with_plugin_daemon_response_stream(
  75. "POST",
  76. f"plugin/{tenant_id}/dispatch/tool/invoke",
  77. ToolInvokeMessage,
  78. data={
  79. "user_id": user_id,
  80. "conversation_id": conversation_id,
  81. "app_id": app_id,
  82. "message_id": message_id,
  83. "data": {
  84. "provider": tool_provider_id.provider_name,
  85. "tool": tool_name,
  86. "credentials": credentials,
  87. "credential_type": credential_type,
  88. "tool_parameters": tool_parameters,
  89. },
  90. },
  91. headers={
  92. "X-Plugin-ID": tool_provider_id.plugin_id,
  93. "Content-Type": "application/json",
  94. },
  95. )
  96. return merge_blob_chunks(response)
  97. def validate_provider_credentials(
  98. self, tenant_id: str, user_id: str, provider: str, credentials: dict[str, Any]
  99. ) -> bool:
  100. """
  101. validate the credentials of the provider
  102. """
  103. tool_provider_id = GenericProviderID(provider)
  104. response = self._request_with_plugin_daemon_response_stream(
  105. "POST",
  106. f"plugin/{tenant_id}/dispatch/tool/validate_credentials",
  107. PluginBasicBooleanResponse,
  108. data={
  109. "user_id": user_id,
  110. "data": {
  111. "provider": tool_provider_id.provider_name,
  112. "credentials": credentials,
  113. },
  114. },
  115. headers={
  116. "X-Plugin-ID": tool_provider_id.plugin_id,
  117. "Content-Type": "application/json",
  118. },
  119. )
  120. for resp in response:
  121. return resp.result
  122. return False
  123. def get_runtime_parameters(
  124. self,
  125. tenant_id: str,
  126. user_id: str,
  127. provider: str,
  128. credentials: dict[str, Any],
  129. tool: str,
  130. conversation_id: Optional[str] = None,
  131. app_id: Optional[str] = None,
  132. message_id: Optional[str] = None,
  133. ) -> list[ToolParameter]:
  134. """
  135. get the runtime parameters of the tool
  136. """
  137. tool_provider_id = GenericProviderID(provider)
  138. class RuntimeParametersResponse(BaseModel):
  139. parameters: list[ToolParameter]
  140. response = self._request_with_plugin_daemon_response_stream(
  141. "POST",
  142. f"plugin/{tenant_id}/dispatch/tool/get_runtime_parameters",
  143. RuntimeParametersResponse,
  144. data={
  145. "user_id": user_id,
  146. "conversation_id": conversation_id,
  147. "app_id": app_id,
  148. "message_id": message_id,
  149. "data": {
  150. "provider": tool_provider_id.provider_name,
  151. "tool": tool,
  152. "credentials": credentials,
  153. },
  154. },
  155. headers={
  156. "X-Plugin-ID": tool_provider_id.plugin_id,
  157. "Content-Type": "application/json",
  158. },
  159. )
  160. for resp in response:
  161. return resp.parameters
  162. return []