plugin.py 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. from collections.abc import Sequence
  2. from pydantic import BaseModel
  3. from core.plugin.entities.bundle import PluginBundleDependency
  4. from core.plugin.entities.plugin import (
  5. GenericProviderID,
  6. PluginDeclaration,
  7. PluginEntity,
  8. PluginInstallation,
  9. PluginInstallationSource,
  10. )
  11. from core.plugin.entities.plugin_daemon import PluginInstallTask, PluginInstallTaskStartResponse, PluginUploadResponse
  12. from core.plugin.manager.base import BasePluginManager
  13. class PluginInstallationManager(BasePluginManager):
  14. def fetch_plugin_by_identifier(
  15. self,
  16. tenant_id: str,
  17. identifier: str,
  18. ) -> bool:
  19. return self._request_with_plugin_daemon_response(
  20. "GET",
  21. f"plugin/{tenant_id}/management/fetch/identifier",
  22. bool,
  23. params={"plugin_unique_identifier": identifier},
  24. )
  25. def list_plugins(self, tenant_id: str) -> list[PluginEntity]:
  26. return self._request_with_plugin_daemon_response(
  27. "GET",
  28. f"plugin/{tenant_id}/management/list",
  29. list[PluginEntity],
  30. params={"page": 1, "page_size": 256},
  31. )
  32. def upload_pkg(
  33. self,
  34. tenant_id: str,
  35. pkg: bytes,
  36. verify_signature: bool = False,
  37. ) -> PluginUploadResponse:
  38. """
  39. Upload a plugin package and return the plugin unique identifier.
  40. """
  41. body = {
  42. "dify_pkg": ("dify_pkg", pkg, "application/octet-stream"),
  43. }
  44. data = {
  45. "verify_signature": "true" if verify_signature else "false",
  46. }
  47. return self._request_with_plugin_daemon_response(
  48. "POST",
  49. f"plugin/{tenant_id}/management/install/upload/package",
  50. PluginUploadResponse,
  51. files=body,
  52. data=data,
  53. )
  54. def upload_bundle(
  55. self,
  56. tenant_id: str,
  57. bundle: bytes,
  58. verify_signature: bool = False,
  59. ) -> Sequence[PluginBundleDependency]:
  60. """
  61. Upload a plugin bundle and return the dependencies.
  62. """
  63. return self._request_with_plugin_daemon_response(
  64. "POST",
  65. f"plugin/{tenant_id}/management/install/upload/bundle",
  66. list[PluginBundleDependency],
  67. files={"dify_bundle": ("dify_bundle", bundle, "application/octet-stream")},
  68. data={"verify_signature": "true" if verify_signature else "false"},
  69. )
  70. def install_from_identifiers(
  71. self, tenant_id: str, identifiers: Sequence[str], source: PluginInstallationSource, meta: dict
  72. ) -> PluginInstallTaskStartResponse:
  73. """
  74. Install a plugin from an identifier.
  75. """
  76. # exception will be raised if the request failed
  77. return self._request_with_plugin_daemon_response(
  78. "POST",
  79. f"plugin/{tenant_id}/management/install/identifiers",
  80. PluginInstallTaskStartResponse,
  81. data={
  82. "plugin_unique_identifiers": identifiers,
  83. "source": source,
  84. "meta": meta,
  85. },
  86. headers={"Content-Type": "application/json"},
  87. )
  88. def fetch_plugin_installation_tasks(self, tenant_id: str, page: int, page_size: int) -> Sequence[PluginInstallTask]:
  89. """
  90. Fetch plugin installation tasks.
  91. """
  92. return self._request_with_plugin_daemon_response(
  93. "GET",
  94. f"plugin/{tenant_id}/management/install/tasks",
  95. list[PluginInstallTask],
  96. params={"page": page, "page_size": page_size},
  97. )
  98. def fetch_plugin_installation_task(self, tenant_id: str, task_id: str) -> PluginInstallTask:
  99. """
  100. Fetch a plugin installation task.
  101. """
  102. return self._request_with_plugin_daemon_response(
  103. "GET",
  104. f"plugin/{tenant_id}/management/install/tasks/{task_id}",
  105. PluginInstallTask,
  106. )
  107. def delete_plugin_installation_task(self, tenant_id: str, task_id: str) -> bool:
  108. """
  109. Delete a plugin installation task.
  110. """
  111. return self._request_with_plugin_daemon_response(
  112. "POST",
  113. f"plugin/{tenant_id}/management/install/tasks/{task_id}/delete",
  114. bool,
  115. )
  116. def delete_all_plugin_installation_task_items(self, tenant_id: str) -> bool:
  117. """
  118. Delete all plugin installation task items.
  119. """
  120. return self._request_with_plugin_daemon_response(
  121. "POST",
  122. f"plugin/{tenant_id}/management/install/tasks/delete_all",
  123. bool,
  124. )
  125. def delete_plugin_installation_task_item(self, tenant_id: str, task_id: str, identifier: str) -> bool:
  126. """
  127. Delete a plugin installation task item.
  128. """
  129. return self._request_with_plugin_daemon_response(
  130. "POST",
  131. f"plugin/{tenant_id}/management/install/tasks/{task_id}/delete/{identifier}",
  132. bool,
  133. )
  134. def fetch_plugin_manifest(self, tenant_id: str, plugin_unique_identifier: str) -> PluginDeclaration:
  135. """
  136. Fetch a plugin manifest.
  137. """
  138. class PluginDeclarationResponse(BaseModel):
  139. declaration: PluginDeclaration
  140. return self._request_with_plugin_daemon_response(
  141. "GET",
  142. f"plugin/{tenant_id}/management/fetch/manifest",
  143. PluginDeclarationResponse,
  144. params={"plugin_unique_identifier": plugin_unique_identifier},
  145. ).declaration
  146. def fetch_plugin_installation_by_ids(
  147. self, tenant_id: str, plugin_ids: Sequence[str]
  148. ) -> Sequence[PluginInstallation]:
  149. """
  150. Fetch plugin installations by ids.
  151. """
  152. return self._request_with_plugin_daemon_response(
  153. "POST",
  154. f"plugin/{tenant_id}/management/installation/fetch/batch",
  155. list[PluginInstallation],
  156. data={"plugin_ids": plugin_ids},
  157. headers={"Content-Type": "application/json"},
  158. )
  159. def fetch_missing_dependencies(self, tenant_id: str, plugin_unique_identifiers: list[str]) -> list[str]:
  160. """
  161. Fetch missing dependencies
  162. """
  163. return self._request_with_plugin_daemon_response(
  164. "POST",
  165. f"plugin/{tenant_id}/management/installation/missing",
  166. list[str],
  167. data={"plugin_unique_identifiers": plugin_unique_identifiers},
  168. headers={"Content-Type": "application/json"},
  169. )
  170. def uninstall(self, tenant_id: str, plugin_installation_id: str) -> bool:
  171. """
  172. Uninstall a plugin.
  173. """
  174. return self._request_with_plugin_daemon_response(
  175. "POST",
  176. f"plugin/{tenant_id}/management/uninstall",
  177. bool,
  178. data={
  179. "plugin_installation_id": plugin_installation_id,
  180. },
  181. headers={"Content-Type": "application/json"},
  182. )
  183. def upgrade_plugin(
  184. self,
  185. tenant_id: str,
  186. original_plugin_unique_identifier: str,
  187. new_plugin_unique_identifier: str,
  188. source: PluginInstallationSource,
  189. meta: dict,
  190. ) -> PluginInstallTaskStartResponse:
  191. """
  192. Upgrade a plugin.
  193. """
  194. return self._request_with_plugin_daemon_response(
  195. "POST",
  196. f"plugin/{tenant_id}/management/install/upgrade",
  197. PluginInstallTaskStartResponse,
  198. data={
  199. "original_plugin_unique_identifier": original_plugin_unique_identifier,
  200. "new_plugin_unique_identifier": new_plugin_unique_identifier,
  201. "source": source,
  202. "meta": meta,
  203. },
  204. headers={"Content-Type": "application/json"},
  205. )
  206. def check_tools_existence(self, tenant_id: str, provider_ids: Sequence[GenericProviderID]) -> Sequence[bool]:
  207. """
  208. Check if the tools exist
  209. """
  210. return self._request_with_plugin_daemon_response(
  211. "POST",
  212. f"plugin/{tenant_id}/management/tools/check_existence",
  213. list[bool],
  214. data={
  215. "provider_ids": [
  216. {
  217. "plugin_id": provider_id.plugin_id,
  218. "provider_name": provider_id.provider_name,
  219. }
  220. for provider_id in provider_ids
  221. ]
  222. },
  223. headers={"Content-Type": "application/json"},
  224. )