plugin.py 15 KB

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