installed_app.py 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. from datetime import datetime, timezone
  2. from flask_login import current_user
  3. from flask_restful import Resource, inputs, marshal_with, reqparse
  4. from sqlalchemy import and_
  5. from werkzeug.exceptions import BadRequest, Forbidden, NotFound
  6. from controllers.console import api
  7. from controllers.console.explore.wraps import InstalledAppResource
  8. from controllers.console.wraps import account_initialization_required, cloud_edition_billing_resource_check
  9. from extensions.ext_database import db
  10. from fields.installed_app_fields import installed_app_list_fields
  11. from libs.login import login_required
  12. from models.model import App, InstalledApp, RecommendedApp
  13. from services.account_service import TenantService
  14. class InstalledAppsListApi(Resource):
  15. @login_required
  16. @account_initialization_required
  17. @marshal_with(installed_app_list_fields)
  18. def get(self):
  19. current_tenant_id = current_user.current_tenant_id
  20. installed_apps = db.session.query(InstalledApp).filter(InstalledApp.tenant_id == current_tenant_id).all()
  21. current_user.role = TenantService.get_user_role(current_user, current_user.current_tenant)
  22. installed_apps = [
  23. {
  24. "id": installed_app.id,
  25. "app": installed_app.app,
  26. "app_owner_tenant_id": installed_app.app_owner_tenant_id,
  27. "is_pinned": installed_app.is_pinned,
  28. "last_used_at": installed_app.last_used_at,
  29. "editable": current_user.role in ["owner", "admin"],
  30. "uninstallable": current_tenant_id == installed_app.app_owner_tenant_id,
  31. }
  32. for installed_app in installed_apps
  33. ]
  34. installed_apps.sort(
  35. key=lambda app: (
  36. -app["is_pinned"],
  37. app["last_used_at"] is None,
  38. -app["last_used_at"].timestamp() if app["last_used_at"] is not None else 0,
  39. )
  40. )
  41. return {"installed_apps": installed_apps}
  42. @login_required
  43. @account_initialization_required
  44. @cloud_edition_billing_resource_check("apps")
  45. def post(self):
  46. parser = reqparse.RequestParser()
  47. parser.add_argument("app_id", type=str, required=True, help="Invalid app_id")
  48. args = parser.parse_args()
  49. recommended_app = RecommendedApp.query.filter(RecommendedApp.app_id == args["app_id"]).first()
  50. if recommended_app is None:
  51. raise NotFound("App not found")
  52. current_tenant_id = current_user.current_tenant_id
  53. app = db.session.query(App).filter(App.id == args["app_id"]).first()
  54. if app is None:
  55. raise NotFound("App not found")
  56. if not app.is_public:
  57. raise Forbidden("You can't install a non-public app")
  58. installed_app = InstalledApp.query.filter(
  59. and_(InstalledApp.app_id == args["app_id"], InstalledApp.tenant_id == current_tenant_id)
  60. ).first()
  61. if installed_app is None:
  62. # todo: position
  63. recommended_app.install_count += 1
  64. new_installed_app = InstalledApp(
  65. app_id=args["app_id"],
  66. tenant_id=current_tenant_id,
  67. app_owner_tenant_id=app.tenant_id,
  68. is_pinned=False,
  69. last_used_at=datetime.now(timezone.utc).replace(tzinfo=None),
  70. )
  71. db.session.add(new_installed_app)
  72. db.session.commit()
  73. return {"message": "App installed successfully"}
  74. class InstalledAppApi(InstalledAppResource):
  75. """
  76. update and delete an installed app
  77. use InstalledAppResource to apply default decorators and get installed_app
  78. """
  79. def delete(self, installed_app):
  80. if installed_app.app_owner_tenant_id == current_user.current_tenant_id:
  81. raise BadRequest("You can't uninstall an app owned by the current tenant")
  82. db.session.delete(installed_app)
  83. db.session.commit()
  84. return {"result": "success", "message": "App uninstalled successfully"}
  85. def patch(self, installed_app):
  86. parser = reqparse.RequestParser()
  87. parser.add_argument("is_pinned", type=inputs.boolean)
  88. args = parser.parse_args()
  89. commit_args = False
  90. if "is_pinned" in args:
  91. installed_app.is_pinned = args["is_pinned"]
  92. commit_args = True
  93. if commit_args:
  94. db.session.commit()
  95. return {"result": "success", "message": "App info updated successfully"}
  96. api.add_resource(InstalledAppsListApi, "/installed-apps")
  97. api.add_resource(InstalledAppApi, "/installed-apps/<uuid:installed_app_id>")