plugin.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  1. import io
  2. from flask import request, send_file
  3. from flask_login import current_user
  4. from flask_restful import Resource, reqparse
  5. from werkzeug.exceptions import Forbidden
  6. from configs import dify_config
  7. from controllers.console import api
  8. from controllers.console.setup import setup_required
  9. from controllers.console.workspace import plugin_permission_required
  10. from controllers.console.wraps import account_initialization_required
  11. from core.model_runtime.utils.encoders import jsonable_encoder
  12. from libs.login import login_required
  13. from models.account import TenantPluginPermission
  14. from services.plugin.plugin_permission_service import PluginPermissionService
  15. from services.plugin.plugin_service import PluginService
  16. class PluginDebuggingKeyApi(Resource):
  17. @setup_required
  18. @login_required
  19. @account_initialization_required
  20. @plugin_permission_required(debug_required=True)
  21. def get(self):
  22. tenant_id = current_user.current_tenant_id
  23. return {
  24. "key": PluginService.get_debugging_key(tenant_id),
  25. "host": dify_config.PLUGIN_REMOTE_INSTALL_HOST,
  26. "port": dify_config.PLUGIN_REMOTE_INSTALL_PORT,
  27. }
  28. class PluginListApi(Resource):
  29. @setup_required
  30. @login_required
  31. @account_initialization_required
  32. def get(self):
  33. tenant_id = current_user.current_tenant_id
  34. plugins = PluginService.list(tenant_id)
  35. return jsonable_encoder({"plugins": plugins})
  36. class PluginIconApi(Resource):
  37. @setup_required
  38. def get(self):
  39. req = reqparse.RequestParser()
  40. req.add_argument("tenant_id", type=str, required=True, location="args")
  41. req.add_argument("filename", type=str, required=True, location="args")
  42. args = req.parse_args()
  43. icon_bytes, mimetype = PluginService.get_asset(args["tenant_id"], args["filename"])
  44. icon_cache_max_age = dify_config.TOOL_ICON_CACHE_MAX_AGE
  45. return send_file(io.BytesIO(icon_bytes), mimetype=mimetype, max_age=icon_cache_max_age)
  46. class PluginUploadFromPkgApi(Resource):
  47. @setup_required
  48. @login_required
  49. @account_initialization_required
  50. @plugin_permission_required(install_required=True)
  51. def post(self):
  52. tenant_id = current_user.current_tenant_id
  53. file = request.files["pkg"]
  54. # check file size
  55. if file.content_length > dify_config.PLUGIN_MAX_PACKAGE_SIZE:
  56. raise ValueError("File size exceeds the maximum allowed size")
  57. content = file.read()
  58. response = PluginService.upload_pkg(tenant_id, content)
  59. return jsonable_encoder(response)
  60. class PluginUploadFromGithubApi(Resource):
  61. @setup_required
  62. @login_required
  63. @account_initialization_required
  64. @plugin_permission_required(install_required=True)
  65. def post(self):
  66. tenant_id = current_user.current_tenant_id
  67. parser = reqparse.RequestParser()
  68. parser.add_argument("repo", type=str, required=True, location="json")
  69. parser.add_argument("version", type=str, required=True, location="json")
  70. parser.add_argument("package", type=str, required=True, location="json")
  71. args = parser.parse_args()
  72. response = PluginService.upload_pkg_from_github(tenant_id, args["repo"], args["version"], args["package"])
  73. return {
  74. "plugin_unique_identifier": response,
  75. }
  76. class PluginInstallFromPkgApi(Resource):
  77. @setup_required
  78. @login_required
  79. @account_initialization_required
  80. @plugin_permission_required(install_required=True)
  81. def post(self):
  82. tenant_id = current_user.current_tenant_id
  83. parser = reqparse.RequestParser()
  84. parser.add_argument("plugin_unique_identifiers", type=list, required=True, location="json")
  85. args = parser.parse_args()
  86. # check if all plugin_unique_identifiers are valid string
  87. for plugin_unique_identifier in args["plugin_unique_identifiers"]:
  88. if not isinstance(plugin_unique_identifier, str):
  89. raise ValueError("Invalid plugin unique identifier")
  90. response = PluginService.install_from_local_pkg(tenant_id, args["plugin_unique_identifiers"])
  91. return jsonable_encoder(response)
  92. class PluginInstallFromGithubApi(Resource):
  93. @setup_required
  94. @login_required
  95. @account_initialization_required
  96. @plugin_permission_required(install_required=True)
  97. def post(self):
  98. tenant_id = current_user.current_tenant_id
  99. parser = reqparse.RequestParser()
  100. parser.add_argument("repo", type=str, required=True, location="json")
  101. parser.add_argument("version", type=str, required=True, location="json")
  102. parser.add_argument("package", type=str, required=True, location="json")
  103. parser.add_argument("plugin_unique_identifier", type=str, required=True, location="json")
  104. args = parser.parse_args()
  105. response = PluginService.install_from_github(
  106. tenant_id,
  107. args["plugin_unique_identifier"],
  108. args["repo"],
  109. args["version"],
  110. args["package"],
  111. )
  112. return jsonable_encoder(response)
  113. class PluginInstallFromMarketplaceApi(Resource):
  114. @setup_required
  115. @login_required
  116. @account_initialization_required
  117. @plugin_permission_required(install_required=True)
  118. def post(self):
  119. tenant_id = current_user.current_tenant_id
  120. parser = reqparse.RequestParser()
  121. parser.add_argument("plugin_unique_identifiers", type=list, required=True, location="json")
  122. args = parser.parse_args()
  123. # check if all plugin_unique_identifiers are valid string
  124. for plugin_unique_identifier in args["plugin_unique_identifiers"]:
  125. if not isinstance(plugin_unique_identifier, str):
  126. raise ValueError("Invalid plugin unique identifier")
  127. response = PluginService.install_from_marketplace_pkg(tenant_id, args["plugin_unique_identifiers"])
  128. return jsonable_encoder(response)
  129. class PluginFetchManifestApi(Resource):
  130. @setup_required
  131. @login_required
  132. @account_initialization_required
  133. @plugin_permission_required(debug_required=True)
  134. def get(self):
  135. tenant_id = current_user.current_tenant_id
  136. parser = reqparse.RequestParser()
  137. parser.add_argument("plugin_unique_identifier", type=str, required=True, location="args")
  138. args = parser.parse_args()
  139. return jsonable_encoder(
  140. {"manifest": PluginService.fetch_plugin_manifest(tenant_id, args["plugin_unique_identifier"]).model_dump()}
  141. )
  142. class PluginFetchInstallTasksApi(Resource):
  143. @setup_required
  144. @login_required
  145. @account_initialization_required
  146. @plugin_permission_required(debug_required=True)
  147. def get(self):
  148. tenant_id = current_user.current_tenant_id
  149. parser = reqparse.RequestParser()
  150. parser.add_argument("page", type=int, required=True, location="args")
  151. parser.add_argument("page_size", type=int, required=True, location="args")
  152. args = parser.parse_args()
  153. return jsonable_encoder(
  154. {"tasks": PluginService.fetch_install_tasks(tenant_id, args["page"], args["page_size"])}
  155. )
  156. class PluginFetchInstallTaskApi(Resource):
  157. @setup_required
  158. @login_required
  159. @account_initialization_required
  160. @plugin_permission_required(debug_required=True)
  161. def get(self, task_id: str):
  162. tenant_id = current_user.current_tenant_id
  163. return jsonable_encoder({"task": PluginService.fetch_install_task(tenant_id, task_id)})
  164. class PluginDeleteInstallTaskApi(Resource):
  165. @setup_required
  166. @login_required
  167. @account_initialization_required
  168. @plugin_permission_required(debug_required=True)
  169. def post(self, task_id: str):
  170. tenant_id = current_user.current_tenant_id
  171. return {"success": PluginService.delete_install_task(tenant_id, task_id)}
  172. class PluginDeleteInstallTaskItemApi(Resource):
  173. @setup_required
  174. @login_required
  175. @account_initialization_required
  176. @plugin_permission_required(debug_required=True)
  177. def post(self, task_id: str, identifier: str):
  178. tenant_id = current_user.current_tenant_id
  179. return {"success": PluginService.delete_install_task_item(tenant_id, task_id, identifier)}
  180. class PluginUpgradeFromMarketplaceApi(Resource):
  181. @setup_required
  182. @login_required
  183. @account_initialization_required
  184. @plugin_permission_required(debug_required=True)
  185. def post(self):
  186. tenant_id = current_user.current_tenant_id
  187. parser = reqparse.RequestParser()
  188. parser.add_argument("original_plugin_unique_identifier", type=str, required=True, location="json")
  189. parser.add_argument("new_plugin_unique_identifier", type=str, required=True, location="json")
  190. args = parser.parse_args()
  191. return jsonable_encoder(
  192. PluginService.upgrade_plugin_with_marketplace(
  193. tenant_id, args["original_plugin_unique_identifier"], args["new_plugin_unique_identifier"]
  194. )
  195. )
  196. class PluginUpgradeFromGithubApi(Resource):
  197. @setup_required
  198. @login_required
  199. @account_initialization_required
  200. @plugin_permission_required(debug_required=True)
  201. def post(self):
  202. tenant_id = current_user.current_tenant_id
  203. parser = reqparse.RequestParser()
  204. parser.add_argument("original_plugin_unique_identifier", type=str, required=True, location="json")
  205. parser.add_argument("new_plugin_unique_identifier", type=str, required=True, location="json")
  206. parser.add_argument("repo", type=str, required=True, location="json")
  207. parser.add_argument("version", type=str, required=True, location="json")
  208. parser.add_argument("package", type=str, required=True, location="json")
  209. args = parser.parse_args()
  210. return jsonable_encoder(
  211. PluginService.upgrade_plugin_with_github(
  212. tenant_id,
  213. args["original_plugin_unique_identifier"],
  214. args["new_plugin_unique_identifier"],
  215. args["repo"],
  216. args["version"],
  217. args["package"],
  218. )
  219. )
  220. class PluginUninstallApi(Resource):
  221. @setup_required
  222. @login_required
  223. @account_initialization_required
  224. @plugin_permission_required(debug_required=True)
  225. def post(self):
  226. req = reqparse.RequestParser()
  227. req.add_argument("plugin_installation_id", type=str, required=True, location="json")
  228. args = req.parse_args()
  229. tenant_id = current_user.current_tenant_id
  230. return {"success": PluginService.uninstall(tenant_id, args["plugin_installation_id"])}
  231. class PluginChangePermissionApi(Resource):
  232. @setup_required
  233. @login_required
  234. @account_initialization_required
  235. def post(self):
  236. user = current_user
  237. if not user.is_admin_or_owner:
  238. raise Forbidden()
  239. req = reqparse.RequestParser()
  240. req.add_argument("install_permission", type=str, required=True, location="json")
  241. req.add_argument("debug_permission", type=str, required=True, location="json")
  242. args = req.parse_args()
  243. install_permission = TenantPluginPermission.InstallPermission(args["install_permission"])
  244. debug_permission = TenantPluginPermission.DebugPermission(args["debug_permission"])
  245. tenant_id = user.current_tenant_id
  246. return {"success": PluginPermissionService.change_permission(tenant_id, install_permission, debug_permission)}
  247. class PluginFetchPermissionApi(Resource):
  248. @setup_required
  249. @login_required
  250. @account_initialization_required
  251. def get(self):
  252. tenant_id = current_user.current_tenant_id
  253. permission = PluginPermissionService.get_permission(tenant_id)
  254. if not permission:
  255. return jsonable_encoder(
  256. {
  257. "install_permission": TenantPluginPermission.InstallPermission.EVERYONE,
  258. "debug_permission": TenantPluginPermission.DebugPermission.EVERYONE,
  259. }
  260. )
  261. return jsonable_encoder(
  262. {
  263. "install_permission": permission.install_permission,
  264. "debug_permission": permission.debug_permission,
  265. }
  266. )
  267. api.add_resource(PluginDebuggingKeyApi, "/workspaces/current/plugin/debugging-key")
  268. api.add_resource(PluginListApi, "/workspaces/current/plugin/list")
  269. api.add_resource(PluginIconApi, "/workspaces/current/plugin/icon")
  270. api.add_resource(PluginUploadFromPkgApi, "/workspaces/current/plugin/upload/pkg")
  271. api.add_resource(PluginUploadFromGithubApi, "/workspaces/current/plugin/upload/github")
  272. api.add_resource(PluginInstallFromPkgApi, "/workspaces/current/plugin/install/pkg")
  273. api.add_resource(PluginInstallFromGithubApi, "/workspaces/current/plugin/install/github")
  274. api.add_resource(PluginUpgradeFromMarketplaceApi, "/workspaces/current/plugin/upgrade/marketplace")
  275. api.add_resource(PluginUpgradeFromGithubApi, "/workspaces/current/plugin/upgrade/github")
  276. api.add_resource(PluginInstallFromMarketplaceApi, "/workspaces/current/plugin/install/marketplace")
  277. api.add_resource(PluginFetchManifestApi, "/workspaces/current/plugin/fetch-manifest")
  278. api.add_resource(PluginFetchInstallTasksApi, "/workspaces/current/plugin/tasks")
  279. api.add_resource(PluginFetchInstallTaskApi, "/workspaces/current/plugin/tasks/<task_id>")
  280. api.add_resource(PluginDeleteInstallTaskApi, "/workspaces/current/plugin/tasks/<task_id>/delete")
  281. api.add_resource(PluginDeleteInstallTaskItemApi, "/workspaces/current/plugin/tasks/<task_id>/delete/<path:identifier>")
  282. api.add_resource(PluginUninstallApi, "/workspaces/current/plugin/uninstall")
  283. api.add_resource(PluginChangePermissionApi, "/workspaces/current/plugin/permission/change")
  284. api.add_resource(PluginFetchPermissionApi, "/workspaces/current/plugin/permission/fetch")