Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

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