Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

plugin.py 19KB

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