installed_app.py 4.6 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 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. if installed_app.app is not None
  34. ]
  35. installed_apps.sort(
  36. key=lambda app: (
  37. -app["is_pinned"],
  38. app["last_used_at"] is None,
  39. -app["last_used_at"].timestamp() if app["last_used_at"] is not None else 0,
  40. )
  41. )
  42. return {"installed_apps": installed_apps}
  43. @login_required
  44. @account_initialization_required
  45. @cloud_edition_billing_resource_check("apps")
  46. def post(self):
  47. parser = reqparse.RequestParser()
  48. parser.add_argument("app_id", type=str, required=True, help="Invalid app_id")
  49. args = parser.parse_args()
  50. recommended_app = RecommendedApp.query.filter(RecommendedApp.app_id == args["app_id"]).first()
  51. if recommended_app is None:
  52. raise NotFound("App not found")
  53. current_tenant_id = current_user.current_tenant_id
  54. app = db.session.query(App).filter(App.id == args["app_id"]).first()
  55. if app is None:
  56. raise NotFound("App not found")
  57. if not app.is_public:
  58. raise Forbidden("You can't install a non-public app")
  59. installed_app = InstalledApp.query.filter(
  60. and_(InstalledApp.app_id == args["app_id"], InstalledApp.tenant_id == current_tenant_id)
  61. ).first()
  62. if installed_app is None:
  63. # todo: position
  64. recommended_app.install_count += 1
  65. new_installed_app = InstalledApp(
  66. app_id=args["app_id"],
  67. tenant_id=current_tenant_id,
  68. app_owner_tenant_id=app.tenant_id,
  69. is_pinned=False,
  70. last_used_at=datetime.now(timezone.utc).replace(tzinfo=None),
  71. )
  72. db.session.add(new_installed_app)
  73. db.session.commit()
  74. return {"message": "App installed successfully"}
  75. class InstalledAppApi(InstalledAppResource):
  76. """
  77. update and delete an installed app
  78. use InstalledAppResource to apply default decorators and get installed_app
  79. """
  80. def delete(self, installed_app):
  81. if installed_app.app_owner_tenant_id == current_user.current_tenant_id:
  82. raise BadRequest("You can't uninstall an app owned by the current tenant")
  83. db.session.delete(installed_app)
  84. db.session.commit()
  85. return {"result": "success", "message": "App uninstalled successfully"}
  86. def patch(self, installed_app):
  87. parser = reqparse.RequestParser()
  88. parser.add_argument("is_pinned", type=inputs.boolean)
  89. args = parser.parse_args()
  90. commit_args = False
  91. if "is_pinned" in args:
  92. installed_app.is_pinned = args["is_pinned"]
  93. commit_args = True
  94. if commit_args:
  95. db.session.commit()
  96. return {"result": "success", "message": "App info updated successfully"}
  97. api.add_resource(InstalledAppsListApi, "/installed-apps")
  98. api.add_resource(InstalledAppApi, "/installed-apps/<uuid:installed_app_id>")