You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

app_import.py 4.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. from typing import cast
  2. from flask_login import current_user
  3. from flask_restx import Resource, marshal_with, reqparse
  4. from sqlalchemy.orm import Session
  5. from werkzeug.exceptions import Forbidden
  6. from controllers.console.app.wraps import get_app_model
  7. from controllers.console.wraps import (
  8. account_initialization_required,
  9. cloud_edition_billing_resource_check,
  10. setup_required,
  11. )
  12. from extensions.ext_database import db
  13. from fields.app_fields import app_import_check_dependencies_fields, app_import_fields
  14. from libs.login import login_required
  15. from models import Account
  16. from models.model import App
  17. from services.app_dsl_service import AppDslService, ImportStatus
  18. from services.enterprise.enterprise_service import EnterpriseService
  19. from services.feature_service import FeatureService
  20. from .. import console_ns
  21. @console_ns.route("/apps/imports")
  22. class AppImportApi(Resource):
  23. @setup_required
  24. @login_required
  25. @account_initialization_required
  26. @marshal_with(app_import_fields)
  27. @cloud_edition_billing_resource_check("apps")
  28. def post(self):
  29. # Check user role first
  30. if not current_user.is_editor:
  31. raise Forbidden()
  32. parser = reqparse.RequestParser()
  33. parser.add_argument("mode", type=str, required=True, location="json")
  34. parser.add_argument("yaml_content", type=str, location="json")
  35. parser.add_argument("yaml_url", type=str, location="json")
  36. parser.add_argument("name", type=str, location="json")
  37. parser.add_argument("description", type=str, location="json")
  38. parser.add_argument("icon_type", type=str, location="json")
  39. parser.add_argument("icon", type=str, location="json")
  40. parser.add_argument("icon_background", type=str, location="json")
  41. parser.add_argument("app_id", type=str, location="json")
  42. args = parser.parse_args()
  43. # Create service with session
  44. with Session(db.engine) as session:
  45. import_service = AppDslService(session)
  46. # Import app
  47. account = cast(Account, current_user)
  48. result = import_service.import_app(
  49. account=account,
  50. import_mode=args["mode"],
  51. yaml_content=args.get("yaml_content"),
  52. yaml_url=args.get("yaml_url"),
  53. name=args.get("name"),
  54. description=args.get("description"),
  55. icon_type=args.get("icon_type"),
  56. icon=args.get("icon"),
  57. icon_background=args.get("icon_background"),
  58. app_id=args.get("app_id"),
  59. )
  60. session.commit()
  61. if result.app_id and FeatureService.get_system_features().webapp_auth.enabled:
  62. # update web app setting as private
  63. EnterpriseService.WebAppAuth.update_app_access_mode(result.app_id, "private")
  64. # Return appropriate status code based on result
  65. status = result.status
  66. if status == ImportStatus.FAILED.value:
  67. return result.model_dump(mode="json"), 400
  68. elif status == ImportStatus.PENDING.value:
  69. return result.model_dump(mode="json"), 202
  70. return result.model_dump(mode="json"), 200
  71. @console_ns.route("/apps/imports/<string:import_id>/confirm")
  72. class AppImportConfirmApi(Resource):
  73. @setup_required
  74. @login_required
  75. @account_initialization_required
  76. @marshal_with(app_import_fields)
  77. def post(self, import_id):
  78. # Check user role first
  79. if not current_user.is_editor:
  80. raise Forbidden()
  81. # Create service with session
  82. with Session(db.engine) as session:
  83. import_service = AppDslService(session)
  84. # Confirm import
  85. account = cast(Account, current_user)
  86. result = import_service.confirm_import(import_id=import_id, account=account)
  87. session.commit()
  88. # Return appropriate status code based on result
  89. if result.status == ImportStatus.FAILED.value:
  90. return result.model_dump(mode="json"), 400
  91. return result.model_dump(mode="json"), 200
  92. @console_ns.route("/apps/imports/<string:app_id>/check-dependencies")
  93. class AppImportCheckDependenciesApi(Resource):
  94. @setup_required
  95. @login_required
  96. @get_app_model
  97. @account_initialization_required
  98. @marshal_with(app_import_check_dependencies_fields)
  99. def get(self, app_model: App):
  100. if not current_user.is_editor:
  101. raise Forbidden()
  102. with Session(db.engine) as session:
  103. import_service = AppDslService(session)
  104. result = import_service.check_dependencies(app_model=app_model)
  105. return result.model_dump(mode="json"), 200