| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- from typing import Optional, Union
-
- from flask_restx import Resource, reqparse
- from pydantic import ValidationError
-
- from controllers.console.app.mcp_server import AppMCPServerStatus
- from controllers.mcp import mcp_ns
- from core.app.app_config.entities import VariableEntity
- from core.mcp import types
- from core.mcp.server.streamable_http import MCPServerStreamableHTTPRequestHandler
- from core.mcp.types import ClientNotification, ClientRequest
- from core.mcp.utils import create_mcp_error_response
- from extensions.ext_database import db
- from libs import helper
- from models.model import App, AppMCPServer, AppMode
-
-
- def int_or_str(value):
- """Validate that a value is either an integer or string."""
- if isinstance(value, (int, str)):
- return value
- else:
- return None
-
-
- # Define parser for both documentation and validation
- mcp_request_parser = reqparse.RequestParser()
- mcp_request_parser.add_argument(
- "jsonrpc", type=str, required=True, location="json", help="JSON-RPC version (should be '2.0')"
- )
- mcp_request_parser.add_argument("method", type=str, required=True, location="json", help="The method to invoke")
- mcp_request_parser.add_argument("params", type=dict, required=False, location="json", help="Parameters for the method")
- mcp_request_parser.add_argument(
- "id", type=int_or_str, required=False, location="json", help="Request ID for tracking responses"
- )
-
-
- @mcp_ns.route("/server/<string:server_code>/mcp")
- class MCPAppApi(Resource):
- @mcp_ns.expect(mcp_request_parser)
- @mcp_ns.doc("handle_mcp_request")
- @mcp_ns.doc(description="Handle Model Context Protocol (MCP) requests for a specific server")
- @mcp_ns.doc(params={"server_code": "Unique identifier for the MCP server"})
- @mcp_ns.doc(
- responses={
- 200: "MCP response successfully processed",
- 400: "Invalid MCP request or parameters",
- 404: "Server or app not found",
- }
- )
- def post(self, server_code: str):
- """Handle MCP requests for a specific server.
-
- Processes JSON-RPC formatted requests according to the Model Context Protocol specification.
- Validates the server status and associated app before processing the request.
-
- Args:
- server_code: Unique identifier for the MCP server
-
- Returns:
- dict: JSON-RPC response from the MCP handler
-
- Raises:
- ValidationError: Invalid request format or parameters
- """
- # Parse and validate all arguments
- args = mcp_request_parser.parse_args()
-
- request_id: Optional[Union[int, str]] = args.get("id")
-
- server = db.session.query(AppMCPServer).where(AppMCPServer.server_code == server_code).first()
- if not server:
- return helper.compact_generate_response(
- create_mcp_error_response(request_id, types.INVALID_REQUEST, "Server Not Found")
- )
-
- if server.status != AppMCPServerStatus.ACTIVE:
- return helper.compact_generate_response(
- create_mcp_error_response(request_id, types.INVALID_REQUEST, "Server is not active")
- )
-
- app = db.session.query(App).where(App.id == server.app_id).first()
- if not app:
- return helper.compact_generate_response(
- create_mcp_error_response(request_id, types.INVALID_REQUEST, "App Not Found")
- )
-
- if app.mode in {AppMode.ADVANCED_CHAT.value, AppMode.WORKFLOW.value}:
- workflow = app.workflow
- if workflow is None:
- return helper.compact_generate_response(
- create_mcp_error_response(request_id, types.INVALID_REQUEST, "App is unavailable")
- )
-
- user_input_form = workflow.user_input_form(to_old_structure=True)
- else:
- app_model_config = app.app_model_config
- if app_model_config is None:
- return helper.compact_generate_response(
- create_mcp_error_response(request_id, types.INVALID_REQUEST, "App is unavailable")
- )
-
- features_dict = app_model_config.to_dict()
- user_input_form = features_dict.get("user_input_form", [])
- converted_user_input_form: list[VariableEntity] = []
- try:
- for item in user_input_form:
- variable_type = item.get("type", "") or list(item.keys())[0]
- variable = item[variable_type]
- converted_user_input_form.append(
- VariableEntity(
- type=variable_type,
- variable=variable.get("variable"),
- description=variable.get("description") or "",
- label=variable.get("label"),
- required=variable.get("required", False),
- max_length=variable.get("max_length"),
- options=variable.get("options") or [],
- )
- )
- except ValidationError as e:
- return helper.compact_generate_response(
- create_mcp_error_response(request_id, types.INVALID_PARAMS, f"Invalid user_input_form: {str(e)}")
- )
-
- try:
- request: ClientRequest | ClientNotification = ClientRequest.model_validate(args)
- except ValidationError as e:
- try:
- notification = ClientNotification.model_validate(args)
- request = notification
- except ValidationError as e:
- return helper.compact_generate_response(
- create_mcp_error_response(request_id, types.INVALID_PARAMS, f"Invalid MCP request: {str(e)}")
- )
-
- mcp_server_handler = MCPServerStreamableHTTPRequestHandler(app, request, converted_user_input_form)
- response = mcp_server_handler.handle()
- return helper.compact_generate_response(response)
|