Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. from flask_restful import Resource, reqparse
  2. from pydantic import ValidationError
  3. from controllers.console.app.mcp_server import AppMCPServerStatus
  4. from controllers.mcp import api
  5. from core.app.app_config.entities import VariableEntity
  6. from core.mcp import types
  7. from core.mcp.server.streamable_http import MCPServerStreamableHTTPRequestHandler
  8. from core.mcp.types import ClientNotification, ClientRequest
  9. from core.mcp.utils import create_mcp_error_response
  10. from extensions.ext_database import db
  11. from libs import helper
  12. from models.model import App, AppMCPServer, AppMode
  13. class MCPAppApi(Resource):
  14. def post(self, server_code):
  15. def int_or_str(value):
  16. if isinstance(value, (int, str)):
  17. return value
  18. else:
  19. return None
  20. parser = reqparse.RequestParser()
  21. parser.add_argument("jsonrpc", type=str, required=True, location="json")
  22. parser.add_argument("method", type=str, required=True, location="json")
  23. parser.add_argument("params", type=dict, required=False, location="json")
  24. parser.add_argument("id", type=int_or_str, required=False, location="json")
  25. args = parser.parse_args()
  26. request_id = args.get("id")
  27. server = db.session.query(AppMCPServer).filter(AppMCPServer.server_code == server_code).first()
  28. if not server:
  29. return helper.compact_generate_response(
  30. create_mcp_error_response(request_id, types.INVALID_REQUEST, "Server Not Found")
  31. )
  32. if server.status != AppMCPServerStatus.ACTIVE:
  33. return helper.compact_generate_response(
  34. create_mcp_error_response(request_id, types.INVALID_REQUEST, "Server is not active")
  35. )
  36. app = db.session.query(App).filter(App.id == server.app_id).first()
  37. if not app:
  38. return helper.compact_generate_response(
  39. create_mcp_error_response(request_id, types.INVALID_REQUEST, "App Not Found")
  40. )
  41. if app.mode in {AppMode.ADVANCED_CHAT.value, AppMode.WORKFLOW.value}:
  42. workflow = app.workflow
  43. if workflow is None:
  44. return helper.compact_generate_response(
  45. create_mcp_error_response(request_id, types.INVALID_REQUEST, "App is unavailable")
  46. )
  47. user_input_form = workflow.user_input_form(to_old_structure=True)
  48. else:
  49. app_model_config = app.app_model_config
  50. if app_model_config is None:
  51. return helper.compact_generate_response(
  52. create_mcp_error_response(request_id, types.INVALID_REQUEST, "App is unavailable")
  53. )
  54. features_dict = app_model_config.to_dict()
  55. user_input_form = features_dict.get("user_input_form", [])
  56. converted_user_input_form: list[VariableEntity] = []
  57. try:
  58. for item in user_input_form:
  59. variable_type = item.get("type", "") or list(item.keys())[0]
  60. variable = item[variable_type]
  61. converted_user_input_form.append(
  62. VariableEntity(
  63. type=variable_type,
  64. variable=variable.get("variable"),
  65. description=variable.get("description") or "",
  66. label=variable.get("label"),
  67. required=variable.get("required", False),
  68. max_length=variable.get("max_length"),
  69. options=variable.get("options") or [],
  70. )
  71. )
  72. except ValidationError as e:
  73. return helper.compact_generate_response(
  74. create_mcp_error_response(request_id, types.INVALID_PARAMS, f"Invalid user_input_form: {str(e)}")
  75. )
  76. try:
  77. request: ClientRequest | ClientNotification = ClientRequest.model_validate(args)
  78. except ValidationError as e:
  79. try:
  80. notification = ClientNotification.model_validate(args)
  81. request = notification
  82. except ValidationError as e:
  83. return helper.compact_generate_response(
  84. create_mcp_error_response(request_id, types.INVALID_PARAMS, f"Invalid MCP request: {str(e)}")
  85. )
  86. mcp_server_handler = MCPServerStreamableHTTPRequestHandler(app, request, converted_user_input_form)
  87. response = mcp_server_handler.handle()
  88. return helper.compact_generate_response(response)
  89. api.add_resource(MCPAppApi, "/server/<string:server_code>/mcp")