plugin.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. import io
  2. from flask import request, send_file
  3. from flask_login import current_user # type: ignore
  4. from flask_restful import Resource, reqparse # type: ignore
  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 core.plugin.manager.exc import PluginDaemonClientSideError
  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. try:
  24. return {
  25. "key": PluginService.get_debugging_key(tenant_id),
  26. "host": dify_config.PLUGIN_REMOTE_INSTALL_HOST,
  27. "port": dify_config.PLUGIN_REMOTE_INSTALL_PORT,
  28. }
  29. except PluginDaemonClientSideError as e:
  30. raise ValueError(e)
  31. class PluginListApi(Resource):
  32. @setup_required
  33. @login_required
  34. @account_initialization_required
  35. def get(self):
  36. tenant_id = current_user.current_tenant_id
  37. try:
  38. plugins = PluginService.list(tenant_id)
  39. except PluginDaemonClientSideError as e:
  40. raise ValueError(e)
  41. return jsonable_encoder({"plugins": plugins})
  42. class PluginListInstallationsFromIdsApi(Resource):
  43. @setup_required
  44. @login_required
  45. @account_initialization_required
  46. def post(self):
  47. tenant_id = current_user.current_tenant_id
  48. parser = reqparse.RequestParser()
  49. parser.add_argument("plugin_ids", type=list, required=True, location="json")
  50. args = parser.parse_args()
  51. try:
  52. plugins = PluginService.list_installations_from_ids(tenant_id, args["plugin_ids"])
  53. except PluginDaemonClientSideError as e:
  54. raise ValueError(e)
  55. return jsonable_encoder({"plugins": plugins})
  56. class PluginIconApi(Resource):
  57. @setup_required
  58. def get(self):
  59. req = reqparse.RequestParser()
  60. req.add_argument("tenant_id", type=str, required=True, location="args")
  61. req.add_argument("filename", type=str, required=True, location="args")
  62. args = req.parse_args()
  63. try:
  64. icon_bytes, mimetype = PluginService.get_asset(args["tenant_id"], args["filename"])
  65. except PluginDaemonClientSideError as e:
  66. raise ValueError(e)
  67. icon_cache_max_age = dify_config.TOOL_ICON_CACHE_MAX_AGE
  68. return send_file(io.BytesIO(icon_bytes), mimetype=mimetype, max_age=icon_cache_max_age)
  69. class PluginUploadFromPkgApi(Resource):
  70. @setup_required
  71. @login_required
  72. @account_initialization_required
  73. @plugin_permission_required(install_required=True)
  74. def post(self):
  75. tenant_id = current_user.current_tenant_id
  76. file = request.files["pkg"]
  77. # check file size
  78. if file.content_length > dify_config.PLUGIN_MAX_PACKAGE_SIZE:
  79. raise ValueError("File size exceeds the maximum allowed size")
  80. content = file.read()
  81. try:
  82. response = PluginService.upload_pkg(tenant_id, content)
  83. except PluginDaemonClientSideError as e:
  84. raise ValueError(e)
  85. return jsonable_encoder(response)
  86. class PluginUploadFromGithubApi(Resource):
  87. @setup_required
  88. @login_required
  89. @account_initialization_required
  90. @plugin_permission_required(install_required=True)
  91. def post(self):
  92. tenant_id = current_user.current_tenant_id
  93. parser = reqparse.RequestParser()
  94. parser.add_argument("repo", type=str, required=True, location="json")
  95. parser.add_argument("version", type=str, required=True, location="json")
  96. parser.add_argument("package", type=str, required=True, location="json")
  97. args = parser.parse_args()
  98. try:
  99. response = PluginService.upload_pkg_from_github(tenant_id, args["repo"], args["version"], args["package"])
  100. except PluginDaemonClientSideError as e:
  101. raise ValueError(e)
  102. return jsonable_encoder(response)
  103. class PluginUploadFromBundleApi(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. file = request.files["bundle"]
  111. # check file size
  112. if file.content_length > dify_config.PLUGIN_MAX_BUNDLE_SIZE:
  113. raise ValueError("File size exceeds the maximum allowed size")
  114. content = file.read()
  115. try:
  116. response = PluginService.upload_bundle(tenant_id, content)
  117. except PluginDaemonClientSideError as e:
  118. raise ValueError(e)
  119. return jsonable_encoder(response)
  120. class PluginInstallFromPkgApi(Resource):
  121. @setup_required
  122. @login_required
  123. @account_initialization_required
  124. @plugin_permission_required(install_required=True)
  125. def post(self):
  126. tenant_id = current_user.current_tenant_id
  127. parser = reqparse.RequestParser()
  128. parser.add_argument("plugin_unique_identifiers", type=list, required=True, location="json")
  129. args = parser.parse_args()
  130. # check if all plugin_unique_identifiers are valid string
  131. for plugin_unique_identifier in args["plugin_unique_identifiers"]:
  132. if not isinstance(plugin_unique_identifier, str):
  133. raise ValueError("Invalid plugin unique identifier")
  134. try:
  135. response = PluginService.install_from_local_pkg(tenant_id, args["plugin_unique_identifiers"])
  136. except PluginDaemonClientSideError as e:
  137. raise ValueError(e)
  138. return jsonable_encoder(response)
  139. class PluginInstallFromGithubApi(Resource):
  140. @setup_required
  141. @login_required
  142. @account_initialization_required
  143. @plugin_permission_required(install_required=True)
  144. def post(self):
  145. tenant_id = current_user.current_tenant_id
  146. parser = reqparse.RequestParser()
  147. parser.add_argument("repo", type=str, required=True, location="json")
  148. parser.add_argument("version", type=str, required=True, location="json")
  149. parser.add_argument("package", type=str, required=True, location="json")
  150. parser.add_argument("plugin_unique_identifier", type=str, required=True, location="json")
  151. args = parser.parse_args()
  152. try:
  153. response = PluginService.install_from_github(
  154. tenant_id,
  155. args["plugin_unique_identifier"],
  156. args["repo"],
  157. args["version"],
  158. args["package"],
  159. )
  160. except PluginDaemonClientSideError as e:
  161. raise ValueError(e)
  162. return jsonable_encoder(response)
  163. class PluginInstallFromMarketplaceApi(Resource):
  164. @setup_required
  165. @login_required
  166. @account_initialization_required
  167. @plugin_permission_required(install_required=True)
  168. def post(self):
  169. tenant_id = current_user.current_tenant_id
  170. parser = reqparse.RequestParser()
  171. parser.add_argument("plugin_unique_identifiers", type=list, required=True, location="json")
  172. args = parser.parse_args()
  173. # check if all plugin_unique_identifiers are valid string
  174. for plugin_unique_identifier in args["plugin_unique_identifiers"]:
  175. if not isinstance(plugin_unique_identifier, str):
  176. raise ValueError("Invalid plugin unique identifier")
  177. try:
  178. response = PluginService.install_from_marketplace_pkg(tenant_id, args["plugin_unique_identifiers"])
  179. except PluginDaemonClientSideError as e:
  180. raise ValueError(e)
  181. return jsonable_encoder(response)
  182. class PluginFetchManifestApi(Resource):
  183. @setup_required
  184. @login_required
  185. @account_initialization_required
  186. @plugin_permission_required(debug_required=True)
  187. def get(self):
  188. tenant_id = current_user.current_tenant_id
  189. parser = reqparse.RequestParser()
  190. parser.add_argument("plugin_unique_identifier", type=str, required=True, location="args")
  191. args = parser.parse_args()
  192. try:
  193. return jsonable_encoder(
  194. {
  195. "manifest": PluginService.fetch_plugin_manifest(
  196. tenant_id, args["plugin_unique_identifier"]
  197. ).model_dump()
  198. }
  199. )
  200. except PluginDaemonClientSideError as e:
  201. raise ValueError(e)
  202. class PluginFetchInstallTasksApi(Resource):
  203. @setup_required
  204. @login_required
  205. @account_initialization_required
  206. @plugin_permission_required(debug_required=True)
  207. def get(self):
  208. tenant_id = current_user.current_tenant_id
  209. parser = reqparse.RequestParser()
  210. parser.add_argument("page", type=int, required=True, location="args")
  211. parser.add_argument("page_size", type=int, required=True, location="args")
  212. args = parser.parse_args()
  213. try:
  214. return jsonable_encoder(
  215. {"tasks": PluginService.fetch_install_tasks(tenant_id, args["page"], args["page_size"])}
  216. )
  217. except PluginDaemonClientSideError as e:
  218. raise ValueError(e)
  219. class PluginFetchInstallTaskApi(Resource):
  220. @setup_required
  221. @login_required
  222. @account_initialization_required
  223. @plugin_permission_required(debug_required=True)
  224. def get(self, task_id: str):
  225. tenant_id = current_user.current_tenant_id
  226. try:
  227. return jsonable_encoder({"task": PluginService.fetch_install_task(tenant_id, task_id)})
  228. except PluginDaemonClientSideError as e:
  229. raise ValueError(e)
  230. class PluginDeleteInstallTaskApi(Resource):
  231. @setup_required
  232. @login_required
  233. @account_initialization_required
  234. @plugin_permission_required(debug_required=True)
  235. def post(self, task_id: str):
  236. tenant_id = current_user.current_tenant_id
  237. try:
  238. return {"success": PluginService.delete_install_task(tenant_id, task_id)}
  239. except PluginDaemonClientSideError as e:
  240. raise ValueError(e)
  241. class PluginDeleteAllInstallTaskItemsApi(Resource):
  242. @setup_required
  243. @login_required
  244. @account_initialization_required
  245. @plugin_permission_required(debug_required=True)
  246. def post(self):
  247. tenant_id = current_user.current_tenant_id
  248. try:
  249. return {"success": PluginService.delete_all_install_task_items(tenant_id)}
  250. except PluginDaemonClientSideError as e:
  251. raise ValueError(e)
  252. class PluginDeleteInstallTaskItemApi(Resource):
  253. @setup_required
  254. @login_required
  255. @account_initialization_required
  256. @plugin_permission_required(debug_required=True)
  257. def post(self, task_id: str, identifier: str):
  258. tenant_id = current_user.current_tenant_id
  259. try:
  260. return {"success": PluginService.delete_install_task_item(tenant_id, task_id, identifier)}
  261. except PluginDaemonClientSideError as e:
  262. raise ValueError(e)
  263. class PluginUpgradeFromMarketplaceApi(Resource):
  264. @setup_required
  265. @login_required
  266. @account_initialization_required
  267. @plugin_permission_required(debug_required=True)
  268. def post(self):
  269. tenant_id = current_user.current_tenant_id
  270. parser = reqparse.RequestParser()
  271. parser.add_argument("original_plugin_unique_identifier", type=str, required=True, location="json")
  272. parser.add_argument("new_plugin_unique_identifier", type=str, required=True, location="json")
  273. args = parser.parse_args()
  274. try:
  275. return jsonable_encoder(
  276. PluginService.upgrade_plugin_with_marketplace(
  277. tenant_id, args["original_plugin_unique_identifier"], args["new_plugin_unique_identifier"]
  278. )
  279. )
  280. except PluginDaemonClientSideError as e:
  281. raise ValueError(e)
  282. class PluginUpgradeFromGithubApi(Resource):
  283. @setup_required
  284. @login_required
  285. @account_initialization_required
  286. @plugin_permission_required(debug_required=True)
  287. def post(self):
  288. tenant_id = current_user.current_tenant_id
  289. parser = reqparse.RequestParser()
  290. parser.add_argument("original_plugin_unique_identifier", type=str, required=True, location="json")
  291. parser.add_argument("new_plugin_unique_identifier", type=str, required=True, location="json")
  292. parser.add_argument("repo", type=str, required=True, location="json")
  293. parser.add_argument("version", type=str, required=True, location="json")
  294. parser.add_argument("package", type=str, required=True, location="json")
  295. args = parser.parse_args()
  296. try:
  297. return jsonable_encoder(
  298. PluginService.upgrade_plugin_with_github(
  299. tenant_id,
  300. args["original_plugin_unique_identifier"],
  301. args["new_plugin_unique_identifier"],
  302. args["repo"],
  303. args["version"],
  304. args["package"],
  305. )
  306. )
  307. except PluginDaemonClientSideError as e:
  308. raise ValueError(e)
  309. class PluginUninstallApi(Resource):
  310. @setup_required
  311. @login_required
  312. @account_initialization_required
  313. @plugin_permission_required(debug_required=True)
  314. def post(self):
  315. req = reqparse.RequestParser()
  316. req.add_argument("plugin_installation_id", type=str, required=True, location="json")
  317. args = req.parse_args()
  318. tenant_id = current_user.current_tenant_id
  319. try:
  320. return {"success": PluginService.uninstall(tenant_id, args["plugin_installation_id"])}
  321. except PluginDaemonClientSideError as e:
  322. raise ValueError(e)
  323. class PluginChangePermissionApi(Resource):
  324. @setup_required
  325. @login_required
  326. @account_initialization_required
  327. def post(self):
  328. user = current_user
  329. if not user.is_admin_or_owner:
  330. raise Forbidden()
  331. req = reqparse.RequestParser()
  332. req.add_argument("install_permission", type=str, required=True, location="json")
  333. req.add_argument("debug_permission", type=str, required=True, location="json")
  334. args = req.parse_args()
  335. install_permission = TenantPluginPermission.InstallPermission(args["install_permission"])
  336. debug_permission = TenantPluginPermission.DebugPermission(args["debug_permission"])
  337. tenant_id = user.current_tenant_id
  338. return {"success": PluginPermissionService.change_permission(tenant_id, install_permission, debug_permission)}
  339. class PluginFetchPermissionApi(Resource):
  340. @setup_required
  341. @login_required
  342. @account_initialization_required
  343. def get(self):
  344. tenant_id = current_user.current_tenant_id
  345. permission = PluginPermissionService.get_permission(tenant_id)
  346. if not permission:
  347. return jsonable_encoder(
  348. {
  349. "install_permission": TenantPluginPermission.InstallPermission.EVERYONE,
  350. "debug_permission": TenantPluginPermission.DebugPermission.EVERYONE,
  351. }
  352. )
  353. return jsonable_encoder(
  354. {
  355. "install_permission": permission.install_permission,
  356. "debug_permission": permission.debug_permission,
  357. }
  358. )
  359. api.add_resource(PluginDebuggingKeyApi, "/workspaces/current/plugin/debugging-key")
  360. api.add_resource(PluginListApi, "/workspaces/current/plugin/list")
  361. api.add_resource(PluginListInstallationsFromIdsApi, "/workspaces/current/plugin/list/installations/ids")
  362. api.add_resource(PluginIconApi, "/workspaces/current/plugin/icon")
  363. api.add_resource(PluginUploadFromPkgApi, "/workspaces/current/plugin/upload/pkg")
  364. api.add_resource(PluginUploadFromGithubApi, "/workspaces/current/plugin/upload/github")
  365. api.add_resource(PluginUploadFromBundleApi, "/workspaces/current/plugin/upload/bundle")
  366. api.add_resource(PluginInstallFromPkgApi, "/workspaces/current/plugin/install/pkg")
  367. api.add_resource(PluginInstallFromGithubApi, "/workspaces/current/plugin/install/github")
  368. api.add_resource(PluginUpgradeFromMarketplaceApi, "/workspaces/current/plugin/upgrade/marketplace")
  369. api.add_resource(PluginUpgradeFromGithubApi, "/workspaces/current/plugin/upgrade/github")
  370. api.add_resource(PluginInstallFromMarketplaceApi, "/workspaces/current/plugin/install/marketplace")
  371. api.add_resource(PluginFetchManifestApi, "/workspaces/current/plugin/fetch-manifest")
  372. api.add_resource(PluginFetchInstallTasksApi, "/workspaces/current/plugin/tasks")
  373. api.add_resource(PluginFetchInstallTaskApi, "/workspaces/current/plugin/tasks/<task_id>")
  374. api.add_resource(PluginDeleteInstallTaskApi, "/workspaces/current/plugin/tasks/<task_id>/delete")
  375. api.add_resource(PluginDeleteAllInstallTaskItemsApi, "/workspaces/current/plugin/tasks/delete_all")
  376. api.add_resource(PluginDeleteInstallTaskItemApi, "/workspaces/current/plugin/tasks/<task_id>/delete/<path:identifier>")
  377. api.add_resource(PluginUninstallApi, "/workspaces/current/plugin/uninstall")
  378. api.add_resource(PluginChangePermissionApi, "/workspaces/current/plugin/permission/change")
  379. api.add_resource(PluginFetchPermissionApi, "/workspaces/current/plugin/permission/fetch")