feat: API docs for mcp (#24424)

Signed-off-by: -LAN- <laipz8200@outlook.com>
This commit is contained in:
-LAN-
2025-08-25 09:27:01 +08:00
committed by GitHub
parent b7466f8b65
commit 3075d415e1
2 changed files with 65 additions and 18 deletions

View File

@@ -1,8 +1,20 @@
from flask import Blueprint from flask import Blueprint
from flask_restx import Namespace
from libs.external_api import ExternalApi from libs.external_api import ExternalApi
bp = Blueprint("mcp", __name__, url_prefix="/mcp") bp = Blueprint("mcp", __name__, url_prefix="/mcp")
api = ExternalApi(bp)
api = ExternalApi(
bp,
version="1.0",
title="MCP API",
description="API for Model Context Protocol operations",
doc="/docs", # Enable Swagger UI at /mcp/docs
)
mcp_ns = Namespace("mcp", description="MCP operations")
from . import mcp from . import mcp
api.add_namespace(mcp_ns)

View File

@@ -1,8 +1,10 @@
from typing import Optional, Union
from flask_restx import Resource, reqparse from flask_restx import Resource, reqparse
from pydantic import ValidationError from pydantic import ValidationError
from controllers.console.app.mcp_server import AppMCPServerStatus from controllers.console.app.mcp_server import AppMCPServerStatus
from controllers.mcp import api from controllers.mcp import mcp_ns
from core.app.app_config.entities import VariableEntity from core.app.app_config.entities import VariableEntity
from core.mcp import types from core.mcp import types
from core.mcp.server.streamable_http import MCPServerStreamableHTTPRequestHandler from core.mcp.server.streamable_http import MCPServerStreamableHTTPRequestHandler
@@ -13,22 +15,58 @@ from libs import helper
from models.model import App, AppMCPServer, AppMode from models.model import App, AppMCPServer, AppMode
class MCPAppApi(Resource): def int_or_str(value):
def post(self, server_code): """Validate that a value is either an integer or string."""
def int_or_str(value):
if isinstance(value, (int, str)): if isinstance(value, (int, str)):
return value return value
else: else:
return None return None
parser = reqparse.RequestParser()
parser.add_argument("jsonrpc", type=str, required=True, location="json")
parser.add_argument("method", type=str, required=True, location="json")
parser.add_argument("params", type=dict, required=False, location="json")
parser.add_argument("id", type=int_or_str, required=False, location="json")
args = parser.parse_args()
request_id = args.get("id") # 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() server = db.session.query(AppMCPServer).where(AppMCPServer.server_code == server_code).first()
if not server: if not server:
@@ -99,6 +137,3 @@ class MCPAppApi(Resource):
mcp_server_handler = MCPServerStreamableHTTPRequestHandler(app, request, converted_user_input_form) mcp_server_handler = MCPServerStreamableHTTPRequestHandler(app, request, converted_user_input_form)
response = mcp_server_handler.handle() response = mcp_server_handler.handle()
return helper.compact_generate_response(response) return helper.compact_generate_response(response)
api.add_resource(MCPAppApi, "/server/<string:server_code>/mcp")