Files
Easy-Agents/easy-agents-mcp/src/main/java/com/easyagents/mcp/client/McpEnvironmentChecker.java
陈子默 43f45956ff feat: 对接 Agent MCP 能力
- 新增 runtime MCP 声明、ClientFactory、Toolkit 适配与工具别名映射

- 增加 MCP 环境检测与 stdio 环境变量透传

- 补齐 MCP 工具事件、审批与生命周期释放测试
2026-05-29 11:08:39 +08:00

319 lines
13 KiB
Java

/*
* Copyright (c) 2023-2026, Easy-Agents (fuhai999@gmail.com).
* <p>
* Licensed under the Apache License, Version 2.0 (the "License");
*/
package com.easyagents.mcp.client;
import com.alibaba.fastjson2.JSON;
import io.modelcontextprotocol.client.McpClient;
import io.modelcontextprotocol.client.McpSyncClient;
import io.modelcontextprotocol.spec.McpSchema;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.nio.charset.StandardCharsets;
import java.time.Duration;
import java.util.HashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.function.Function;
/**
* MCP 配置与运行环境检测器。
*/
public class McpEnvironmentChecker {
private static final Logger log = LoggerFactory.getLogger(McpEnvironmentChecker.class);
private static final Duration COMMAND_TIMEOUT = Duration.ofSeconds(3);
private static final Duration MCP_REQUEST_TIMEOUT = Duration.ofSeconds(10);
private static final Set<String> SUPPORTED_TRANSPORTS = Set.of("stdio", "http-sse", "http-stream");
private static final Set<String> KNOWN_VERSION_COMMANDS = Set.of(
"node", "npm", "npx", "pnpm", "python", "python3", "pip", "pip3");
private final boolean probeEnabled;
private final Function<String, McpTransportFactory> transportFactoryProvider;
/**
* 创建启用连接探测的检测器。
*/
public McpEnvironmentChecker() {
this(true);
}
/**
* 创建可控制连接探测行为的检测器。
*
* @param probeEnabled 是否启用 MCP 连接探测
*/
McpEnvironmentChecker(boolean probeEnabled) {
this(probeEnabled, McpEnvironmentChecker::defaultTransportFactory);
}
/**
* 创建可注入 transport 工厂的检测器。
*
* @param probeEnabled 是否启用 MCP 连接探测
* @param transportFactoryProvider transport 工厂提供器
*/
McpEnvironmentChecker(boolean probeEnabled,
Function<String, McpTransportFactory> transportFactoryProvider) {
this.probeEnabled = probeEnabled;
this.transportFactoryProvider = transportFactoryProvider;
}
/**
* 检测 MCP 配置。
*
* @param configJson MCP 配置 JSON
* @return 检测结果
*/
public McpEnvironmentCheckResult check(String configJson) {
McpEnvironmentCheckResult result = new McpEnvironmentCheckResult();
if (configJson == null || configJson.isBlank()) {
result.setOverallStatus(McpCheckStatus.FAILED);
result.addServer(failedServer("config", null, "configJson", "MCP 配置 JSON 不能为空", null));
return result;
}
McpConfig config;
try {
config = JSON.parseObject(configJson, McpConfig.class);
} catch (Exception error) {
result.setOverallStatus(McpCheckStatus.FAILED);
result.addServer(failedServer("config", null, "json", "MCP 配置 JSON 格式错误", sanitize(error)));
return result;
}
if (config == null || config.getMcpServers() == null || config.getMcpServers().isEmpty()) {
result.setOverallStatus(McpCheckStatus.FAILED);
result.addServer(failedServer("config", null, "mcpServers", "mcpServers 不能为空", null));
return result;
}
for (Map.Entry<String, McpConfig.ServerSpec> entry : config.getMcpServers().entrySet()) {
result.addServer(checkServer(entry.getKey(), entry.getValue()));
}
return result;
}
private McpServerCheckResult checkServer(String serverName, McpConfig.ServerSpec spec) {
McpServerCheckResult result = new McpServerCheckResult();
result.setServerName(serverName);
result.setTransport(transport(spec));
if (serverName == null || serverName.isBlank()) {
result.addCheck(McpCheckItem.of("serverName", McpCheckStatus.FAILED,
"MCP 服务名称不能为空", null));
}
if (spec == null) {
result.addCheck(McpCheckItem.of("server", McpCheckStatus.FAILED,
"MCP 服务配置不能为空", null));
return result;
}
String transport = transport(spec);
result.setTransport(transport);
if (!SUPPORTED_TRANSPORTS.contains(transport)) {
result.addCheck(McpCheckItem.of("transport", McpCheckStatus.FAILED,
"不支持的 MCP 传输类型", transport));
return result;
}
result.addCheck(McpCheckItem.of("transport", McpCheckStatus.SUCCESS,
"MCP 传输类型可用", transport));
Map<String, String> resolvedEnv = resolveEnv(spec.getEnv(), result);
if ("stdio".equals(transport)) {
validateStdio(spec, result);
} else {
validateHttp(spec, result);
}
if (probeEnabled && result.getStatus() != McpCheckStatus.FAILED) {
probe(serverName, spec, resolvedEnv, result);
}
return result;
}
private void validateStdio(McpConfig.ServerSpec spec, McpServerCheckResult result) {
if (spec.getCommand() == null || spec.getCommand().isBlank()) {
result.addCheck(McpCheckItem.of("command", McpCheckStatus.FAILED,
"stdio MCP 必须配置 command", null));
return;
}
result.addCheck(checkCommand(spec.getCommand()));
}
private void validateHttp(McpConfig.ServerSpec spec, McpServerCheckResult result) {
if (spec.getUrl() == null || spec.getUrl().isBlank()) {
result.addCheck(McpCheckItem.of("url", McpCheckStatus.FAILED,
"HTTP MCP 必须配置 url", null));
return;
}
result.addCheck(McpCheckItem.of("url", McpCheckStatus.SUCCESS,
"MCP 连接地址已配置", spec.getUrl()));
}
private McpCheckItem checkCommand(String command) {
String executable = executableName(command);
boolean known = KNOWN_VERSION_COMMANDS.contains(executable);
ProcessBuilder builder = known
? new ProcessBuilder(command, "--version")
: new ProcessBuilder(command);
try {
Process process = builder.redirectErrorStream(true).start();
boolean finished = process.waitFor(COMMAND_TIMEOUT.toMillis(), TimeUnit.MILLISECONDS);
if (!finished) {
process.destroyForcibly();
return McpCheckItem.of("command", McpCheckStatus.SUCCESS,
command + " 可启动", "版本检测超时,已终止检测进程");
}
String output = new String(process.getInputStream().readAllBytes(), StandardCharsets.UTF_8).trim();
if (process.exitValue() == 0 || !known) {
return McpCheckItem.of("command", McpCheckStatus.SUCCESS,
command + " 可用", firstLine(output));
}
return McpCheckItem.of("command", McpCheckStatus.WARNING,
command + " 可启动但返回非零状态", firstLine(output));
} catch (Exception error) {
return McpCheckItem.of("command", McpCheckStatus.FAILED,
"容器内未找到命令:" + command, sanitize(error));
}
}
private void probe(String serverName,
McpConfig.ServerSpec spec,
Map<String, String> resolvedEnv,
McpServerCheckResult result) {
CloseableTransport transport = null;
McpSyncClient client = null;
try {
transport = transportFactoryProvider.apply(spec.getTransport()).create(spec, resolvedEnv);
client = McpClient.sync(transport.getTransport())
.requestTimeout(MCP_REQUEST_TIMEOUT)
.build();
client.initialize();
McpSchema.ListToolsResult toolsResult = client.listTools();
int toolCount = toolsResult == null || toolsResult.tools() == null ? 0 : toolsResult.tools().size();
result.setToolCount(toolCount);
if (toolCount == 0) {
result.addCheck(McpCheckItem.of("tools", McpCheckStatus.WARNING,
"MCP 已连接,但没有发现工具", null));
} else {
result.addCheck(McpCheckItem.of("tools", McpCheckStatus.SUCCESS,
"MCP 工具列表获取成功", String.valueOf(toolCount)));
}
} catch (Exception error) {
log.debug("MCP check failed for server: {}", serverName, error);
result.addCheck(McpCheckItem.of("connection", McpCheckStatus.FAILED,
"MCP 初始化或工具发现失败", sanitize(error)));
} finally {
closeQuietly(client);
closeQuietly(transport);
}
}
private Map<String, String> resolveEnv(Map<String, String> env, McpServerCheckResult result) {
Map<String, String> resolved = new HashMap<>();
if (env == null || env.isEmpty()) {
result.addCheck(McpCheckItem.of("env", McpCheckStatus.SUCCESS,
"未配置额外环境变量", null));
return resolved;
}
for (Map.Entry<String, String> entry : env.entrySet()) {
String key = entry.getKey();
String value = entry.getValue();
if (value != null && value.startsWith("${input:") && value.endsWith("}")) {
String inputId = value.substring("${input:".length(), value.length() - 1);
String resolvedValue = System.getProperty("mcp.input." + inputId);
if (resolvedValue == null || resolvedValue.isBlank()) {
result.addCheck(McpCheckItem.of("env", McpCheckStatus.FAILED,
"环境变量未解析:" + key, "input:" + inputId));
continue;
}
resolved.put(key, resolvedValue);
result.addCheck(McpCheckItem.of("env", McpCheckStatus.SUCCESS,
"环境变量已解析:" + key, "input:" + inputId));
continue;
}
resolved.put(key, value);
result.addCheck(McpCheckItem.of("env", McpCheckStatus.SUCCESS,
"环境变量已配置:" + key, null));
}
return resolved;
}
private McpServerCheckResult failedServer(String serverName,
String transport,
String name,
String message,
String detail) {
McpServerCheckResult server = new McpServerCheckResult();
server.setServerName(serverName);
server.setTransport(transport);
server.addCheck(McpCheckItem.of(name, McpCheckStatus.FAILED, message, detail));
return server;
}
private static McpTransportFactory defaultTransportFactory(String transportType) {
return switch (transport(transportType)) {
case "stdio" -> new StdioTransportFactory();
case "http-sse" -> new HttpSseTransportFactory();
case "http-stream" -> new HttpStreamTransportFactory();
default -> throw new IllegalArgumentException("Unsupported transport: " + transportType);
};
}
private String transport(McpConfig.ServerSpec spec) {
return spec == null ? "stdio" : transport(spec.getTransport());
}
private static String transport(String value) {
return value == null || value.isBlank() ? "stdio" : value.toLowerCase(Locale.ROOT);
}
private String executableName(String command) {
int slash = Math.max(command.lastIndexOf('/'), command.lastIndexOf('\\'));
String name = slash >= 0 ? command.substring(slash + 1) : command;
return name.endsWith(".cmd") ? name.substring(0, name.length() - 4) : name;
}
private String firstLine(String value) {
if (value == null || value.isBlank()) {
return null;
}
int lineEnd = value.indexOf('\n');
String line = lineEnd >= 0 ? value.substring(0, lineEnd) : value;
return sanitize(line.trim());
}
private String sanitize(Throwable error) {
if (error == null) {
return null;
}
String message = error.getMessage();
return message == null || message.isBlank() ? error.getClass().getSimpleName() : sanitize(message);
}
private String sanitize(String value) {
if (value == null) {
return null;
}
String sanitized = value.replaceAll("(?i)(api[_-]?key|token|secret|password)=([^\\s,;]+)", "$1=******");
return sanitized.length() > 500 ? sanitized.substring(0, 500) : sanitized;
}
private void closeQuietly(AutoCloseable closeable) {
if (closeable == null) {
return;
}
try {
closeable.close();
} catch (Exception error) {
log.debug("Failed to close MCP check resource.", error);
}
}
}