| | | 1 | | using Shared.Contracts.Mcp; |
| | | 2 | | |
| | | 3 | | namespace UIBlazor.Services.Settings; |
| | | 4 | | |
| | | 5 | | public class McpSettingsProvider( |
| | | 6 | | ILocalStorageService storage, |
| | | 7 | | ILogger<McpSettingsProvider> logger, |
| | | 8 | | IVsBridge vsBridge, |
| | | 9 | | HttpClient httpClient) |
| | 4 | 10 | | : BaseSettingsProvider<McpOptions>(storage, logger, "McpSettings"), IMcpSettingsProvider |
| | | 11 | | { |
| | | 12 | | public async Task StopAllAsync() |
| | | 13 | | { |
| | 0 | 14 | | await vsBridge.ExecuteToolAsync(BasicEnum.McpStopAll); |
| | 0 | 15 | | } |
| | | 16 | | |
| | | 17 | | public override async Task ResetAsync() |
| | | 18 | | { |
| | 1 | 19 | | Current.Enabled = true; |
| | 1 | 20 | | Current.Servers = []; |
| | 1 | 21 | | Current.ServerApprovalModes = []; |
| | 1 | 22 | | Current.ServerErrors = []; |
| | 1 | 23 | | Current.ServerEnabledStates = []; |
| | 1 | 24 | | Current.ToolDisabledStates = []; |
| | 1 | 25 | | await SaveAsync(); |
| | 1 | 26 | | } |
| | | 27 | | |
| | | 28 | | protected override async Task AfterInitAsync() |
| | | 29 | | { |
| | 0 | 30 | | _ = LoadMcpFileAsync(); |
| | 0 | 31 | | } |
| | | 32 | | |
| | | 33 | | /// <summary> |
| | | 34 | | /// Load servers from %APPDATA%\Agent\mcp.json via VsBridge |
| | | 35 | | /// </summary> |
| | | 36 | | public async Task LoadMcpFileAsync() |
| | | 37 | | { |
| | | 38 | | try |
| | | 39 | | { |
| | 2 | 40 | | logger.LogInformation("Loading MCP settings from mcp.json"); |
| | 2 | 41 | | Current.ServerErrors.Clear(); |
| | 2 | 42 | | var result = await vsBridge.ExecuteToolAsync(BasicEnum.ReadMcpSettingsFile); |
| | | 43 | | |
| | | 44 | | #if DEBUG |
| | | 45 | | result = HeadlessMocker.GetVsToolResult(result); |
| | | 46 | | #endif |
| | | 47 | | |
| | 2 | 48 | | if (!result.Success || string.IsNullOrEmpty(result.Result)) |
| | | 49 | | { |
| | 1 | 50 | | logger.LogWarning(result.Success ? "mcp.json is empty" : $"Failed to read mcp.json: {result.ErrorMessage |
| | 1 | 51 | | return; |
| | | 52 | | } |
| | | 53 | | |
| | 1 | 54 | | var settingsFile = JsonUtils.Deserialize<McpSettingsFile>(result.Result); |
| | 1 | 55 | | if (settingsFile?.McpServers == null) |
| | | 56 | | { |
| | 0 | 57 | | logger.LogWarning("mcp.json has no servers defined"); |
| | 0 | 58 | | return; |
| | | 59 | | } |
| | | 60 | | |
| | 1 | 61 | | var servers = new List<McpServerConfig>(); |
| | 1 | 62 | | var initServerTasks = new List<Task>(); |
| | 4 | 63 | | foreach (var (name, entry) in settingsFile.McpServers) |
| | | 64 | | { |
| | 1 | 65 | | logger.LogInformation($"Loading server: {name}"); |
| | 1 | 66 | | var isRemote = !string.IsNullOrEmpty(entry.Url); |
| | 1 | 67 | | var server = new McpServerConfig |
| | 1 | 68 | | { |
| | 1 | 69 | | Name = name, |
| | 1 | 70 | | Transport = isRemote ? "http" : "stdio", |
| | 1 | 71 | | Command = entry.Command ?? string.Empty, |
| | 1 | 72 | | Args = entry.Args ?? [], |
| | 1 | 73 | | Url = entry.Url ?? string.Empty, |
| | 1 | 74 | | Endpoint = entry.Url ?? string.Empty, |
| | 1 | 75 | | Env = entry.Env ?? [], |
| | 1 | 76 | | Enabled = true |
| | 1 | 77 | | }; |
| | | 78 | | |
| | 1 | 79 | | servers.Add(server); |
| | 1 | 80 | | initServerTasks.Add(InitToolsAsync(server)); |
| | | 81 | | } |
| | | 82 | | |
| | 1 | 83 | | await Task.WhenAll(initServerTasks); |
| | 1 | 84 | | Current.Servers = servers; |
| | 1 | 85 | | logger.LogInformation($"MCP settings loaded: {servers.Count} servers"); |
| | 1 | 86 | | await SaveAsync(); |
| | 1 | 87 | | } |
| | 0 | 88 | | catch (Exception ex) |
| | | 89 | | { |
| | 0 | 90 | | logger.LogError($"Error loading MCP settings: {ex.Message}"); |
| | 0 | 91 | | Current.ServerErrors["__global__"] = ex.Message; |
| | 0 | 92 | | } |
| | 2 | 93 | | } |
| | | 94 | | |
| | | 95 | | private async Task InitToolsAsync(McpServerConfig server) |
| | | 96 | | { |
| | 1 | 97 | | if (server.Tools.Count == 0) |
| | | 98 | | { |
| | 1 | 99 | | var toolsResult = await RefreshToolsAsync(server); |
| | 1 | 100 | | if (!toolsResult.StartsWith("Success")) |
| | | 101 | | { |
| | 0 | 102 | | logger.LogError($"Failed to load tools for server {server.Name}: {toolsResult}"); |
| | 0 | 103 | | Current.ServerErrors[server.Name] = toolsResult; |
| | 0 | 104 | | server.Enabled = false; |
| | | 105 | | } |
| | | 106 | | else |
| | | 107 | | { |
| | 1 | 108 | | logger.LogInformation($"Loaded tools for server {server.Name}: {toolsResult}"); |
| | 1 | 109 | | Current.ServerErrors.Remove(server.Name); |
| | | 110 | | } |
| | | 111 | | } |
| | 1 | 112 | | } |
| | | 113 | | |
| | | 114 | | /// <summary> |
| | | 115 | | /// Открыть mcp.json в редакторе VS |
| | | 116 | | /// </summary> |
| | | 117 | | public async Task OpenSettingsFileAsync() |
| | | 118 | | { |
| | 0 | 119 | | await vsBridge.ExecuteToolAsync(BasicEnum.OpenMcpSettings); |
| | 0 | 120 | | } |
| | | 121 | | |
| | | 122 | | public async Task<string> RefreshToolsAsync(McpServerConfig server) |
| | | 123 | | { |
| | 1 | 124 | | var updateResult = $"Error refreshing tools for {server.Name}."; |
| | | 125 | | try |
| | | 126 | | { |
| | 1 | 127 | | logger.LogInformation($"Refreshing tools for server: {server.Name} ({server.Transport})"); |
| | | 128 | | |
| | 1 | 129 | | if (server.Transport == "stdio") |
| | | 130 | | { |
| | 1 | 131 | | var argsString = string.Join(" ", server.Args); |
| | 1 | 132 | | var toolArgs = new Dictionary<string, object> |
| | 1 | 133 | | { |
| | 1 | 134 | | { "serverId", server.Name }, |
| | 1 | 135 | | { "command", server.Command }, |
| | 1 | 136 | | { "args", argsString }, |
| | 1 | 137 | | { "env", server.Env } |
| | 1 | 138 | | }; |
| | | 139 | | |
| | 1 | 140 | | logger.LogInformation($"Starting stdio server: {server.Command} {argsString}"); |
| | 1 | 141 | | var result = await vsBridge.ExecuteToolAsync(BasicEnum.McpGetTools, toolArgs); |
| | | 142 | | #if DEBUG |
| | | 143 | | result = HeadlessMocker.GetVsToolResult(result); |
| | | 144 | | #endif |
| | 1 | 145 | | if (!result.Success) |
| | | 146 | | { |
| | 0 | 147 | | logger.LogError($"Failed to get tools from {server.Name}: {result.ErrorMessage}"); |
| | 0 | 148 | | return $"Error: {result.ErrorMessage}"; |
| | | 149 | | } |
| | | 150 | | |
| | 1 | 151 | | var mcpData = JsonUtils.Deserialize<JsonElement>(result.Result); |
| | 1 | 152 | | updateResult = await UpdateServerToolsAsync(server, mcpData); |
| | 1 | 153 | | logger.LogInformation($"Refresh result for {server.Name}: {updateResult}"); |
| | | 154 | | } |
| | | 155 | | else // http sse |
| | | 156 | | { |
| | 0 | 157 | | logger.LogInformation($"Connecting to HTTP MCP server: {server.Url}"); |
| | | 158 | | // MCP SSE handshake |
| | 0 | 159 | | using var request = new HttpRequestMessage(HttpMethod.Get, server.Url); |
| | 0 | 160 | | using var handshakeResponse = await httpClient.SendAsync(request, HttpCompletionOption.ResponseHeadersRe |
| | 0 | 161 | | if (!handshakeResponse.IsSuccessStatusCode) |
| | | 162 | | { |
| | 0 | 163 | | logger.LogError($"HTTP server {server.Name} returned status: {handshakeResponse.StatusCode}"); |
| | 0 | 164 | | return $"Error: HTTP {handshakeResponse.StatusCode}"; |
| | | 165 | | } |
| | 0 | 166 | | var stream = await handshakeResponse.Content.ReadAsStreamAsync(); |
| | 0 | 167 | | using var reader = new StreamReader(stream); |
| | | 168 | | |
| | 0 | 169 | | var postUrl = string.Empty; |
| | | 170 | | |
| | | 171 | | while (true) |
| | | 172 | | { |
| | 0 | 173 | | var line = await reader.ReadLineAsync(); |
| | 0 | 174 | | if (line == null) break; |
| | | 175 | | |
| | 0 | 176 | | if (line.StartsWith("data: ") && !line.Contains('{')) |
| | | 177 | | { |
| | 0 | 178 | | var path = line[6..].Trim(); |
| | 0 | 179 | | var baseUri = new Uri(server.Url); |
| | 0 | 180 | | postUrl = new Uri(baseUri, path).ToString(); |
| | 0 | 181 | | logger.LogInformation($"HTTP MCP endpoint: {postUrl}"); |
| | | 182 | | |
| | 0 | 183 | | var nextLine = await reader.ReadLineAsync(); |
| | 0 | 184 | | if (nextLine != null && nextLine.StartsWith("event: endpoint")) |
| | | 185 | | { |
| | | 186 | | break; |
| | | 187 | | } |
| | | 188 | | } |
| | | 189 | | } |
| | | 190 | | |
| | 0 | 191 | | var mcpRequest = new McpRequest |
| | 0 | 192 | | { |
| | 0 | 193 | | Method = "tools/list", |
| | 0 | 194 | | Params = new { }, |
| | 0 | 195 | | Id = "list_req_" + Guid.NewGuid().ToString("N") |
| | 0 | 196 | | }; |
| | | 197 | | |
| | 0 | 198 | | using var requestMessage = new HttpRequestMessage(HttpMethod.Post, postUrl) |
| | 0 | 199 | | { |
| | 0 | 200 | | Content = new StringContent(JsonUtils.Serialize(mcpRequest), Encoding.UTF8, "application/json") |
| | 0 | 201 | | }; |
| | | 202 | | |
| | 0 | 203 | | requestMessage.Content.Headers.ContentType!.CharSet = null; |
| | 0 | 204 | | logger.LogInformation($"Requesting tools list from {postUrl}"); |
| | 0 | 205 | | var postResponse = await httpClient.SendAsync(requestMessage); |
| | | 206 | | |
| | 0 | 207 | | if (!postResponse.IsSuccessStatusCode) |
| | | 208 | | { |
| | 0 | 209 | | logger.LogError($"HTTP server {server.Name} returned code: {postResponse.StatusCode}"); |
| | 0 | 210 | | return $"{updateResult} {postResponse.StatusCode}"; |
| | | 211 | | } |
| | | 212 | | |
| | | 213 | | while (true) |
| | | 214 | | { |
| | 0 | 215 | | var line = await reader.ReadLineAsync(); |
| | 0 | 216 | | if (line == null) |
| | | 217 | | break; |
| | | 218 | | |
| | 0 | 219 | | if (line.StartsWith("data: ")) |
| | | 220 | | { |
| | 0 | 221 | | var jsonData = line[6..].Trim(); |
| | 0 | 222 | | var mcpData = JsonUtils.Deserialize<McpResponse>(jsonData); |
| | 0 | 223 | | if (mcpData?.Id == mcpRequest.Id && mcpData.Result is JsonElement jsonElement) |
| | | 224 | | { |
| | 0 | 225 | | updateResult = await UpdateServerToolsAsync(server, jsonElement); |
| | 0 | 226 | | logger.LogInformation($"HTTP refresh result for {server.Name}: {updateResult}"); |
| | 0 | 227 | | break; |
| | | 228 | | } |
| | | 229 | | } |
| | | 230 | | } |
| | 0 | 231 | | } |
| | 1 | 232 | | } |
| | 0 | 233 | | catch (Exception ex) |
| | | 234 | | { |
| | 0 | 235 | | logger.LogError($"Error refreshing tools for {server.Name}: {ex.Message}"); |
| | 0 | 236 | | return $"Error: {ex.Message}"; |
| | | 237 | | } |
| | | 238 | | |
| | | 239 | | // Restore tool enabled state from persisted settings |
| | 1 | 240 | | if (server.Tools.Count > 0) |
| | | 241 | | { |
| | 4 | 242 | | foreach (var tool in server.Tools) |
| | | 243 | | { |
| | 1 | 244 | | var toolKey = $"{server.Name}:{tool.Name}"; |
| | 1 | 245 | | tool.Enabled = !Current.ToolDisabledStates.Contains(toolKey); |
| | | 246 | | } |
| | | 247 | | } |
| | | 248 | | |
| | 1 | 249 | | return updateResult; |
| | 1 | 250 | | } |
| | | 251 | | |
| | | 252 | | private async Task<string> UpdateServerToolsAsync(McpServerConfig server, JsonElement resultElement) |
| | | 253 | | { |
| | 1 | 254 | | var listResult = resultElement.GetObject<McpListToolsResult>(); |
| | 1 | 255 | | if (listResult == null) |
| | | 256 | | { |
| | 0 | 257 | | logger.LogError($"No tools found in response for {server.Name}"); |
| | 0 | 258 | | return "Error: Not found tools"; |
| | | 259 | | } |
| | | 260 | | |
| | 2 | 261 | | var newTools = listResult.Tools.Select(t => new McpToolConfig |
| | 2 | 262 | | { |
| | 2 | 263 | | Name = t.Name, |
| | 2 | 264 | | Description = t.Description, |
| | 2 | 265 | | InputSchema = t.InputSchema as JsonElement?, |
| | 2 | 266 | | RequiredArguments = ExtractRequiredArguments(t.InputSchema) |
| | 2 | 267 | | }).ToList(); |
| | | 268 | | |
| | 1 | 269 | | logger.LogInformation($"Updating {server.Name} with {newTools.Count} tools"); |
| | | 270 | | |
| | | 271 | | // Merge with existing tools (preserve enabled state) |
| | 4 | 272 | | foreach (var tool in newTools) |
| | | 273 | | { |
| | 1 | 274 | | var toolKey = $"{server.Name}:{tool.Name}"; |
| | 1 | 275 | | tool.Enabled = !Current.ToolDisabledStates.Contains(toolKey); |
| | | 276 | | } |
| | | 277 | | |
| | 1 | 278 | | server.Tools = newTools; |
| | 1 | 279 | | await SaveAsync(); |
| | 1 | 280 | | return $"Success: Found {newTools.Count} tools"; |
| | 1 | 281 | | } |
| | | 282 | | |
| | | 283 | | private static List<string> ExtractRequiredArguments(object? rawSchema) |
| | | 284 | | { |
| | 1 | 285 | | var required = new List<string>(); |
| | 1 | 286 | | if (rawSchema is not JsonElement { ValueKind: JsonValueKind.Object } jsonElement || |
| | 1 | 287 | | !jsonElement.TryGetProperty("required", out var requiredProp) || |
| | 1 | 288 | | requiredProp.ValueKind != JsonValueKind.Array) |
| | 1 | 289 | | return required; |
| | | 290 | | |
| | 0 | 291 | | foreach (var item in requiredProp.EnumerateArray()) |
| | | 292 | | { |
| | 0 | 293 | | if (item.ValueKind == JsonValueKind.String) |
| | | 294 | | { |
| | 0 | 295 | | required.Add(item.GetString()!); |
| | | 296 | | } |
| | | 297 | | } |
| | 0 | 298 | | return required; |
| | | 299 | | } |
| | | 300 | | } |