plugin.py 14 KB

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