313 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			313 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						||
 | 
						||
namespace app\api\controller\mcp;
 | 
						||
 | 
						||
class Mcp extends Base
 | 
						||
{
 | 
						||
    // 工具定义常量
 | 
						||
    private const ADD_FUNCTION = [
 | 
						||
        [
 | 
						||
        'name' => 'add',
 | 
						||
        'description' => "计算两个数字的和",
 | 
						||
        'inputSchema' => [
 | 
						||
            'type' => 'object',
 | 
						||
            'properties' => [
 | 
						||
                'a' => [
 | 
						||
                    'description' => '第一个加数(必填)',
 | 
						||
                    'example' => 10,
 | 
						||
                    'type' => 'number',
 | 
						||
                ],
 | 
						||
                'b' => [
 | 
						||
                    'description' => '第二个加数(必填)',
 | 
						||
                    'example' => 20,
 | 
						||
                    'type' => 'number',
 | 
						||
                ]
 | 
						||
            ]
 | 
						||
          ]
 | 
						||
        ],
 | 
						||
        [
 | 
						||
            'name' => 'sub',
 | 
						||
            'description' => "计算两个数字的减法",
 | 
						||
            'inputSchema' => [
 | 
						||
                'type' => 'object',
 | 
						||
                'properties' => [
 | 
						||
                    'a' => [
 | 
						||
                        'description' => '第一个减数(必填)',
 | 
						||
                        'example' => 10,
 | 
						||
                        'type' => 'number',
 | 
						||
                    ],
 | 
						||
                    'b' => [
 | 
						||
                        'description' => '第二个减数(必填)',
 | 
						||
                        'example' => 20,
 | 
						||
                        'type' => 'number',
 | 
						||
                    ]
 | 
						||
                ]
 | 
						||
            ]
 | 
						||
        ],
 | 
						||
    ];
 | 
						||
 | 
						||
 | 
						||
    public function sse()
 | 
						||
    {
 | 
						||
        // 设置 CORS 头
 | 
						||
        header('Access-Control-Allow-Origin: *');
 | 
						||
        header('Access-Control-Allow-Methods: POST, OPTIONS');
 | 
						||
        header('Access-Control-Allow-Headers: Content-Type, Authorization');
 | 
						||
 | 
						||
        // 处理 OPTIONS 预检请求
 | 
						||
        if ($_SERVER['REQUEST_METHOD'] === 'OPTIONS') {
 | 
						||
            http_response_code(200);
 | 
						||
            exit;
 | 
						||
        }
 | 
						||
 | 
						||
        // 设置 SSE 响应头
 | 
						||
        header('Content-Type: text/event-stream');
 | 
						||
        header('Cache-Control: no-cache');
 | 
						||
        header('Connection: keep-alive');
 | 
						||
        header('X-Accel-Buffering: no');
 | 
						||
 | 
						||
        // 获取原始输入
 | 
						||
        $input = file_get_contents('php://input');
 | 
						||
        file_put_contents('mcp_debug.log', date('Y-m-d H:i:s').' Raw input: '.$input.PHP_EOL, FILE_APPEND);
 | 
						||
 | 
						||
        // 解析 JSON-RPC 请求
 | 
						||
        $request = json_decode($input, true);
 | 
						||
 | 
						||
        // 处理请求
 | 
						||
        if (isset($request['method'])) {
 | 
						||
            switch($request['method']) {
 | 
						||
                case 'initialize':
 | 
						||
                    $this->handleInitializeRequest($request);
 | 
						||
                    break;
 | 
						||
                case 'function':
 | 
						||
                case 'tools/call': // 新增支持tools/call方法
 | 
						||
                    $this->handleFunctionRequest($request);
 | 
						||
                    break;
 | 
						||
                case 'ping':
 | 
						||
                    $this->sendJsonRpcResponse($request['id'] ?? null, ['result' => 'pong']);
 | 
						||
                    break;
 | 
						||
                case 'notifications/initialized':
 | 
						||
                    $this->sendJsonRpcResponse($request['id'] ?? null, ['result' => 'success']);
 | 
						||
                    break;
 | 
						||
                case 'tools/list':
 | 
						||
                    $this->sendJsonRpcResponse($request['id'] ?? null, ['tools' => self::ADD_FUNCTION]);
 | 
						||
                    break;
 | 
						||
                default:
 | 
						||
                    $this->handleOtherRequests($request);
 | 
						||
            }
 | 
						||
        } else {
 | 
						||
            $this->handleOtherRequests($request);
 | 
						||
        }
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 处理初始化请求
 | 
						||
     */
 | 
						||
    private function handleInitializeRequest($request)
 | 
						||
    {
 | 
						||
        $id = $request['id'] ?? null;
 | 
						||
 | 
						||
        // 记录完整的初始化请求内容
 | 
						||
        file_put_contents('mcp_debug.log', "Received initialize request: " . json_encode($request) . PHP_EOL, FILE_APPEND);
 | 
						||
 | 
						||
        // 发送初始化响应
 | 
						||
        $this->sendJsonRpcResponse($id, [
 | 
						||
            'capabilities' => [
 | 
						||
                'progress' => true,
 | 
						||
                'functions' => self::ADD_FUNCTION
 | 
						||
            ],
 | 
						||
            'serverInfo' => [
 | 
						||
                'name' => 'Calculator Server',
 | 
						||
                'version' => '1.0.0',
 | 
						||
                'vendor' => 'MyCompany',
 | 
						||
                'license' => 'MIT'
 | 
						||
            ],
 | 
						||
            'protocolVersion' => '2025-03-26'
 | 
						||
        ]);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 处理函数调用请求
 | 
						||
     */
 | 
						||
    public function handleFunctionRequest($request)
 | 
						||
    {
 | 
						||
        $id = $request['id'] ?? null;
 | 
						||
        $function = $request['params']['function'] ?? '';
 | 
						||
        $arguments = $request['params']['arguments'] ?? [];
 | 
						||
 | 
						||
//        // 恢复函数名验证
 | 
						||
//        if ($function !== 'add') {
 | 
						||
//            $this->sendJsonRpcResponse($id, [
 | 
						||
//                'error' => [
 | 
						||
//                    'code' => -32601,
 | 
						||
//                    'message' => 'Unsupported function',
 | 
						||
//                    'data' => "Function not found: $function"
 | 
						||
//                ]
 | 
						||
//            ]);
 | 
						||
//            return;
 | 
						||
//        }
 | 
						||
 | 
						||
//        // 提取参数验证为独立方法
 | 
						||
//        $validationResult = $this->validateAddParameters($arguments, $id);
 | 
						||
//        if ($validationResult !== true) {
 | 
						||
//            $this->sendJsonRpcResponse($id, $validationResult);
 | 
						||
//            return;
 | 
						||
//        }
 | 
						||
 | 
						||
        // 解构参数
 | 
						||
        $a = $arguments['a'] ?? 0;
 | 
						||
        $b = $arguments['b'] ?? 0;
 | 
						||
 | 
						||
        // 发送进度通知
 | 
						||
        $this->sendNotification('progress', ['id' => $id, 'message' => '正在计算...']);
 | 
						||
 | 
						||
        // 模拟处理延迟
 | 
						||
        sleep(1);
 | 
						||
 | 
						||
        $result = $this->calculateAdd($a, $b);
 | 
						||
        $this->sendJsonRpcResponse($id, [
 | 
						||
//            '' => [
 | 
						||
                'operation' => 'add',
 | 
						||
                'result' => $result,
 | 
						||
                'expression' => "$a + $b = $result",
 | 
						||
                'timestamp' => time()
 | 
						||
//            ]
 | 
						||
        ]);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 验证加法参数
 | 
						||
     * @param array $arguments
 | 
						||
     * @param mixed $id
 | 
						||
     * @return array|bool 返回true表示验证通过,否则返回错误响应数组
 | 
						||
     */
 | 
						||
    private function validateAddParameters(array $arguments, $id)
 | 
						||
    {
 | 
						||
        if (!isset($arguments['a'], $arguments['b'])) {
 | 
						||
            return [
 | 
						||
                'error' => [
 | 
						||
                    'code' => -32602,
 | 
						||
                    'message' => 'Invalid params',
 | 
						||
                    'data' => 'Missing required parameters: a and b'
 | 
						||
                ]
 | 
						||
            ];
 | 
						||
        }
 | 
						||
 | 
						||
        if (!is_numeric($arguments['a']) || !is_numeric($arguments['b'])) {
 | 
						||
            return [
 | 
						||
                'error' => [
 | 
						||
                    'code' => -32602,
 | 
						||
                    'message' => 'Invalid params',
 | 
						||
                    'data' => 'Parameters must be numbers'
 | 
						||
                ]
 | 
						||
            ];
 | 
						||
        }
 | 
						||
 | 
						||
        return true;
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 处理其他类型的请求
 | 
						||
     */
 | 
						||
    private function handleOtherRequests($request)
 | 
						||
    {
 | 
						||
        if (!$request || !isset($request['function'])) {
 | 
						||
            $this->sendJsonRpcResponse($request['id'] ?? null, ['error' => 'Invalid request format']);
 | 
						||
            return;
 | 
						||
        }
 | 
						||
 | 
						||
        if ($request['function'] === 'add' || $request['function'] === 'tools/call') {
 | 
						||
            $a = $request['params']['a'] ?? 0;
 | 
						||
            $b = $request['params']['b'] ?? 0;
 | 
						||
 | 
						||
            if (!is_numeric($a) || !is_numeric($b)) {
 | 
						||
                $this->sendEvent('error', 'Parameters must be numbers');
 | 
						||
                return;
 | 
						||
            }
 | 
						||
 | 
						||
            // 模拟处理延迟
 | 
						||
            sleep(1);
 | 
						||
 | 
						||
            $result = $this->calculateAdd($a, $b);
 | 
						||
            $this->sendEvent('result', [
 | 
						||
                'operation' => 'add',
 | 
						||
                'result' => $result,
 | 
						||
                'expression' => "$a + $b = $result"
 | 
						||
            ]);
 | 
						||
        } else {
 | 
						||
            $this->sendEvent('error', 'Unsupported function: ' . $request['function']);
 | 
						||
        }
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 执行加法计算
 | 
						||
     */
 | 
						||
    private function calculateAdd(float $a, float $b): float
 | 
						||
    {
 | 
						||
        return $a + $b;
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 发送 JSON-RPC 响应
 | 
						||
     */
 | 
						||
    private function sendJsonRpcResponse($id, $resultOrError)
 | 
						||
    {
 | 
						||
        $response = [
 | 
						||
            'jsonrpc' => '2.0',
 | 
						||
            'id' => $id
 | 
						||
        ];
 | 
						||
 | 
						||
        if (isset($resultOrError['error'])) {
 | 
						||
            $response['error'] = $resultOrError['error'];
 | 
						||
        } else {
 | 
						||
            $response['result'] = $resultOrError;
 | 
						||
        }
 | 
						||
 | 
						||
        $this->sendStreamData(json_encode($response, JSON_UNESCAPED_UNICODE));
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 发送通知(用于进度更新等)
 | 
						||
     */
 | 
						||
    private function sendNotification($method, $params)
 | 
						||
    {
 | 
						||
        $notification = [
 | 
						||
            'jsonrpc' => '2.0',
 | 
						||
            'method' => $method,
 | 
						||
            'params' => $params
 | 
						||
        ];
 | 
						||
 | 
						||
        $this->sendStreamData(json_encode($notification, JSON_UNESCAPED_UNICODE));
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 发送标准 SSE 事件
 | 
						||
     */
 | 
						||
    private function sendEvent($type, $data)
 | 
						||
    {
 | 
						||
        $jsonData = json_encode($data, JSON_UNESCAPED_UNICODE);
 | 
						||
        echo "event: $type\n";
 | 
						||
        echo "data: $jsonData\n\n";
 | 
						||
        $this->flushBuffers();
 | 
						||
        file_put_contents('mcp_debug.log', "Sent event: $type - $jsonData".PHP_EOL, FILE_APPEND);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 发送流数据
 | 
						||
     */
 | 
						||
    private function sendStreamData(string $data)
 | 
						||
    {
 | 
						||
        echo "data: $data\n\n";
 | 
						||
        $this->flushBuffers();
 | 
						||
        file_put_contents('mcp_debug.log', "JSON-RPC Response: $data".PHP_EOL, FILE_APPEND);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * 刷新输出缓冲区
 | 
						||
     */
 | 
						||
    private function flushBuffers()
 | 
						||
    {
 | 
						||
        ob_flush();
 | 
						||
        flush();
 | 
						||
    }
 | 
						||
} |