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.py 4.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. import logging
  2. from flask import request
  3. from flask_restx import Resource, marshal_with, reqparse
  4. from werkzeug.exceptions import Unauthorized
  5. from controllers.common import fields
  6. from controllers.web import api
  7. from controllers.web.error import AppUnavailableError
  8. from controllers.web.wraps import WebApiResource
  9. from core.app.app_config.common.parameters_mapping import get_parameters_from_feature_dict
  10. from libs.passport import PassportService
  11. from models.model import App, AppMode
  12. from services.app_service import AppService
  13. from services.enterprise.enterprise_service import EnterpriseService
  14. from services.feature_service import FeatureService
  15. from services.webapp_auth_service import WebAppAuthService
  16. logger = logging.getLogger(__name__)
  17. class AppParameterApi(WebApiResource):
  18. """Resource for app variables."""
  19. @marshal_with(fields.parameters_fields)
  20. def get(self, app_model: App, end_user):
  21. """Retrieve app parameters."""
  22. if app_model.mode in {AppMode.ADVANCED_CHAT.value, AppMode.WORKFLOW.value}:
  23. workflow = app_model.workflow
  24. if workflow is None:
  25. raise AppUnavailableError()
  26. features_dict = workflow.features_dict
  27. user_input_form = workflow.user_input_form(to_old_structure=True)
  28. else:
  29. app_model_config = app_model.app_model_config
  30. if app_model_config is None:
  31. raise AppUnavailableError()
  32. features_dict = app_model_config.to_dict()
  33. user_input_form = features_dict.get("user_input_form", [])
  34. return get_parameters_from_feature_dict(features_dict=features_dict, user_input_form=user_input_form)
  35. class AppMeta(WebApiResource):
  36. def get(self, app_model: App, end_user):
  37. """Get app meta"""
  38. return AppService().get_app_meta(app_model)
  39. class AppAccessMode(Resource):
  40. def get(self):
  41. parser = reqparse.RequestParser()
  42. parser.add_argument("appId", type=str, required=False, location="args")
  43. parser.add_argument("appCode", type=str, required=False, location="args")
  44. args = parser.parse_args()
  45. features = FeatureService.get_system_features()
  46. if not features.webapp_auth.enabled:
  47. return {"accessMode": "public"}
  48. app_id = args.get("appId")
  49. if args.get("appCode"):
  50. app_code = args["appCode"]
  51. app_id = AppService.get_app_id_by_code(app_code)
  52. if not app_id:
  53. raise ValueError("appId or appCode must be provided")
  54. res = EnterpriseService.WebAppAuth.get_app_access_mode_by_id(app_id)
  55. return {"accessMode": res.access_mode}
  56. class AppWebAuthPermission(Resource):
  57. def get(self):
  58. user_id = "visitor"
  59. try:
  60. auth_header = request.headers.get("Authorization")
  61. if auth_header is None:
  62. raise Unauthorized("Authorization header is missing.")
  63. if " " not in auth_header:
  64. raise Unauthorized("Invalid Authorization header format. Expected 'Bearer <api-key>' format.")
  65. auth_scheme, tk = auth_header.split(None, 1)
  66. auth_scheme = auth_scheme.lower()
  67. if auth_scheme != "bearer":
  68. raise Unauthorized("Authorization scheme must be 'Bearer'")
  69. decoded = PassportService().verify(tk)
  70. user_id = decoded.get("user_id", "visitor")
  71. except Unauthorized:
  72. raise
  73. except Exception:
  74. logger.exception("Unexpected error during auth verification")
  75. raise
  76. features = FeatureService.get_system_features()
  77. if not features.webapp_auth.enabled:
  78. return {"result": True}
  79. parser = reqparse.RequestParser()
  80. parser.add_argument("appId", type=str, required=True, location="args")
  81. args = parser.parse_args()
  82. app_id = args["appId"]
  83. app_code = AppService.get_app_code_by_id(app_id)
  84. res = True
  85. if WebAppAuthService.is_app_require_permission_check(app_id=app_id):
  86. res = EnterpriseService.WebAppAuth.is_user_allowed_to_access_webapp(str(user_id), app_code)
  87. return {"result": res}
  88. api.add_resource(AppParameterApi, "/parameters")
  89. api.add_resource(AppMeta, "/meta")
  90. # webapp auth apis
  91. api.add_resource(AppAccessMode, "/webapp/access-mode")
  92. api.add_resource(AppWebAuthPermission, "/webapp/permission")