diff --git a/ModelContextProtocol.sln b/ModelContextProtocol.sln index 064dc40d..1ceb3a23 100644 --- a/ModelContextProtocol.sln +++ b/ModelContextProtocol.sln @@ -50,6 +50,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "QuickstartWeatherServer", " EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "QuickstartClient", "samples\QuickstartClient\QuickstartClient.csproj", "{0D1552DC-E6ED-4AAC-5562-12F8352F46AA}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EverythingServer", "samples\EverythingServer\EverythingServer.csproj", "{17B8453F-AB72-99C5-E5EA-D0B065A6AE65}" +EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ModelContextProtocol.AspNetCore", "src\ModelContextProtocol.AspNetCore\ModelContextProtocol.AspNetCore.csproj", "{37B6A5E0-9995-497D-8B43-3BC6870CC716}" EndProject Global @@ -94,6 +96,10 @@ Global {0D1552DC-E6ED-4AAC-5562-12F8352F46AA}.Debug|Any CPU.Build.0 = Debug|Any CPU {0D1552DC-E6ED-4AAC-5562-12F8352F46AA}.Release|Any CPU.ActiveCfg = Release|Any CPU {0D1552DC-E6ED-4AAC-5562-12F8352F46AA}.Release|Any CPU.Build.0 = Release|Any CPU + {17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Debug|Any CPU.Build.0 = Debug|Any CPU + {17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Release|Any CPU.ActiveCfg = Release|Any CPU + {17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Release|Any CPU.Build.0 = Release|Any CPU {37B6A5E0-9995-497D-8B43-3BC6870CC716}.Debug|Any CPU.ActiveCfg = Debug|Any CPU {37B6A5E0-9995-497D-8B43-3BC6870CC716}.Debug|Any CPU.Build.0 = Debug|Any CPU {37B6A5E0-9995-497D-8B43-3BC6870CC716}.Release|Any CPU.ActiveCfg = Release|Any CPU @@ -113,6 +119,7 @@ Global {0C6D0512-D26D-63D3-5019-C5F7A657B28C} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8} {4653EB0C-8FC0-98F4-E9C8-220EDA7A69DF} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8} {0D1552DC-E6ED-4AAC-5562-12F8352F46AA} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8} + {17B8453F-AB72-99C5-E5EA-D0B065A6AE65} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8} {37B6A5E0-9995-497D-8B43-3BC6870CC716} = {A2F1F52A-9107-4BF8-8C3F-2F6670E7D0AD} EndGlobalSection GlobalSection(ExtensibilityGlobals) = postSolution diff --git a/samples/EverythingServer/EverythingServer.csproj b/samples/EverythingServer/EverythingServer.csproj new file mode 100644 index 00000000..3aee2bc2 --- /dev/null +++ b/samples/EverythingServer/EverythingServer.csproj @@ -0,0 +1,18 @@ + + + + net9.0 + enable + enable + Exe + + + + + + + + + + + diff --git a/samples/EverythingServer/LoggingUpdateMessageSender.cs b/samples/EverythingServer/LoggingUpdateMessageSender.cs new file mode 100644 index 00000000..7b64aa2c --- /dev/null +++ b/samples/EverythingServer/LoggingUpdateMessageSender.cs @@ -0,0 +1,43 @@ +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using ModelContextProtocol; +using ModelContextProtocol.Protocol.Types; +using ModelContextProtocol.Server; + +namespace EverythingServer; + +public class LoggingUpdateMessageSender(IMcpServer server, Func getMinLevel) : BackgroundService +{ + readonly Dictionary _loggingLevelMap = new() + { + { LoggingLevel.Debug, "Debug-level message" }, + { LoggingLevel.Info, "Info-level message" }, + { LoggingLevel.Notice, "Notice-level message" }, + { LoggingLevel.Warning, "Warning-level message" }, + { LoggingLevel.Error, "Error-level message" }, + { LoggingLevel.Critical, "Critical-level message" }, + { LoggingLevel.Alert, "Alert-level message" }, + { LoggingLevel.Emergency, "Emergency-level message" } + }; + + protected override async Task ExecuteAsync(CancellationToken stoppingToken) + { + while (!stoppingToken.IsCancellationRequested) + { + var newLevel = (LoggingLevel)Random.Shared.Next(_loggingLevelMap.Count); + + var message = new + { + Level = newLevel.ToString().ToLower(), + Data = _loggingLevelMap[newLevel], + }; + + if (newLevel > getMinLevel()) + { + await server.SendNotificationAsync("notifications/message", message, cancellationToken: stoppingToken); + } + + await Task.Delay(15000, stoppingToken); + } + } +} \ No newline at end of file diff --git a/samples/EverythingServer/Program.cs b/samples/EverythingServer/Program.cs new file mode 100644 index 00000000..17ee0753 --- /dev/null +++ b/samples/EverythingServer/Program.cs @@ -0,0 +1,194 @@ +using EverythingServer; +using EverythingServer.Prompts; +using EverythingServer.Tools; +using Microsoft.Extensions.AI; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using Microsoft.Extensions.Logging; +using ModelContextProtocol; +using ModelContextProtocol.Protocol.Types; +using ModelContextProtocol.Server; + +var builder = Host.CreateApplicationBuilder(args); +builder.Logging.AddConsole(consoleLogOptions => +{ + // Configure all logs to go to stderr + consoleLogOptions.LogToStandardErrorThreshold = LogLevel.Trace; +}); + +HashSet subscriptions = []; +var _minimumLoggingLevel = LoggingLevel.Debug; + +builder.Services + .AddMcpServer() + .WithStdioServerTransport() + .WithTools() + .WithTools() + .WithTools() + .WithTools() + .WithTools() + .WithTools() + .WithTools() + .WithPrompts() + .WithPrompts() + .WithListResourceTemplatesHandler((ctx, ct) => + { + return Task.FromResult(new ListResourceTemplatesResult + { + ResourceTemplates = + [ + new ResourceTemplate { Name = "Static Resource", Description = "A static resource with a numeric ID", UriTemplate = "test://static/resource/{id}" } + ] + }); + }) + .WithReadResourceHandler((ctx, ct) => + { + var uri = ctx.Params?.Uri; + + if (uri is null || !uri.StartsWith("test://static/resource/")) + { + throw new NotSupportedException($"Unknown resource: {uri}"); + } + + int index = int.Parse(uri["test://static/resource/".Length..]) - 1; + + if (index < 0 || index >= ResourceGenerator.Resources.Count) + { + throw new NotSupportedException($"Unknown resource: {uri}"); + } + + var resource = ResourceGenerator.Resources[index]; + + if (resource.MimeType == "text/plain") + { + return Task.FromResult(new ReadResourceResult + { + Contents = [new TextResourceContents + { + Text = resource.Description!, + MimeType = resource.MimeType, + Uri = resource.Uri, + }] + }); + } + else + { + return Task.FromResult(new ReadResourceResult + { + Contents = [new BlobResourceContents + { + Blob = resource.Description!, + MimeType = resource.MimeType, + Uri = resource.Uri, + }] + }); + } + }) + .WithSubscribeToResourcesHandler(async (ctx, ct) => + { + var uri = ctx.Params?.Uri; + + if (uri is not null) + { + subscriptions.Add(uri); + + await ctx.Server.RequestSamplingAsync([ + new ChatMessage(ChatRole.System, "You are a helpful test server"), + new ChatMessage(ChatRole.User, $"Resource {uri}, context: A new subscription was started"), + ], + options: new ChatOptions + { + MaxOutputTokens = 100, + Temperature = 0.7f, + }, + cancellationToken: ct); + } + + return new EmptyResult(); + }) + .WithUnsubscribeFromResourcesHandler((ctx, ct) => + { + var uri = ctx.Params?.Uri; + if (uri is not null) + { + subscriptions.Remove(uri); + } + return Task.FromResult(new EmptyResult()); + }) + .WithGetCompletionHandler((ctx, ct) => + { + var exampleCompletions = new Dictionary> + { + { "style", ["casual", "formal", "technical", "friendly"] }, + { "temperature", ["0", "0.5", "0.7", "1.0"] }, + { "resourceId", ["1", "2", "3", "4", "5"] } + }; + + if (ctx.Params is not { } @params) + { + throw new NotSupportedException($"Params are required."); + } + + var @ref = @params.Ref; + var argument = @params.Argument; + + if (@ref.Type == "ref/resource") + { + var resourceId = @ref.Uri?.Split("/").Last(); + + if (resourceId is null) + { + return Task.FromResult(new CompleteResult()); + } + + var values = exampleCompletions["resourceId"].Where(id => id.StartsWith(argument.Value)); + + return Task.FromResult(new CompleteResult + { + Completion = new Completion { Values = [..values], HasMore = false, Total = values.Count() } + }); + } + + if (@ref.Type == "ref/prompt") + { + if (!exampleCompletions.TryGetValue(argument.Name, out IEnumerable? value)) + { + throw new NotSupportedException($"Unknown argument name: {argument.Name}"); + } + + var values = value.Where(value => value.StartsWith(argument.Value)); + return Task.FromResult(new CompleteResult + { + Completion = new Completion { Values = [..values], HasMore = false, Total = values.Count() } + }); + } + + throw new NotSupportedException($"Unknown reference type: {@ref.Type}"); + }) + .WithSetLoggingLevelHandler(async (ctx, ct) => + { + if (ctx.Params?.Level is null) + { + throw new McpException("Missing required argument 'level'"); + } + + _minimumLoggingLevel = ctx.Params.Level; + + await ctx.Server.SendNotificationAsync("notifications/message", new + { + Level = "debug", + Logger = "test-server", + Data = $"Logging level set to {_minimumLoggingLevel}", + }, cancellationToken: ct); + + return new EmptyResult(); + }) + ; + +builder.Services.AddSingleton(subscriptions); +builder.Services.AddHostedService(); +builder.Services.AddHostedService(); + +builder.Services.AddSingleton>(_ => () => _minimumLoggingLevel); + +await builder.Build().RunAsync(); diff --git a/samples/EverythingServer/Prompts/ComplexPromptType.cs b/samples/EverythingServer/Prompts/ComplexPromptType.cs new file mode 100644 index 00000000..8b47a07e --- /dev/null +++ b/samples/EverythingServer/Prompts/ComplexPromptType.cs @@ -0,0 +1,22 @@ +using EverythingServer.Tools; +using Microsoft.Extensions.AI; +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Prompts; + +[McpServerPromptType] +public class ComplexPromptType +{ + [McpServerPrompt(Name = "complex_prompt"), Description("A prompt with arguments")] + public static IEnumerable ComplexPrompt( + [Description("Temperature setting")] int temperature, + [Description("Output style")] string? style = null) + { + return [ + new ChatMessage(ChatRole.User,$"This is a complex prompt with arguments: temperature={temperature}, style={style}"), + new ChatMessage(ChatRole.Assistant, "I understand. You've provided a complex prompt with temperature and style arguments. How would you like me to proceed?"), + new ChatMessage(ChatRole.User, [new DataContent(TinyImageTool.MCP_TINY_IMAGE)]) + ]; + } +} diff --git a/samples/EverythingServer/Prompts/SimplePromptType.cs b/samples/EverythingServer/Prompts/SimplePromptType.cs new file mode 100644 index 00000000..d6ba51a3 --- /dev/null +++ b/samples/EverythingServer/Prompts/SimplePromptType.cs @@ -0,0 +1,11 @@ +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Prompts; + +[McpServerPromptType] +public class SimplePromptType +{ + [McpServerPrompt(Name = "simple_prompt"), Description("A prompt without arguments")] + public static string SimplePrompt() => "This is a simple prompt without arguments"; +} diff --git a/samples/EverythingServer/ResourceGenerator.cs b/samples/EverythingServer/ResourceGenerator.cs new file mode 100644 index 00000000..54764b8c --- /dev/null +++ b/samples/EverythingServer/ResourceGenerator.cs @@ -0,0 +1,37 @@ +using ModelContextProtocol.Protocol.Types; +using System; +using System.Collections.Generic; +using System.Linq; + +namespace EverythingServer; + +static class ResourceGenerator +{ + private static readonly List _resources = Enumerable.Range(1, 100).Select(i => + { + var uri = $"test://static/resource/{i}"; + if (i % 2 != 0) + { + return new Resource + { + Uri = uri, + Name = $"Resource {i}", + MimeType = "text/plain", + Description = $"Resource {i}: This is a plaintext resource" + }; + } + else + { + var buffer = System.Text.Encoding.UTF8.GetBytes($"Resource {i}: This is a base64 blob"); + return new Resource + { + Uri = uri, + Name = $"Resource {i}", + MimeType = "application/octet-stream", + Description = Convert.ToBase64String(buffer) + }; + } + }).ToList(); + + public static IReadOnlyList Resources => _resources; +} \ No newline at end of file diff --git a/samples/EverythingServer/SubscriptionMessageSender.cs b/samples/EverythingServer/SubscriptionMessageSender.cs new file mode 100644 index 00000000..774d9852 --- /dev/null +++ b/samples/EverythingServer/SubscriptionMessageSender.cs @@ -0,0 +1,23 @@ +using Microsoft.Extensions.Hosting; +using ModelContextProtocol; +using ModelContextProtocol.Server; + +internal class SubscriptionMessageSender(IMcpServer server, HashSet subscriptions) : BackgroundService +{ + protected override async Task ExecuteAsync(CancellationToken stoppingToken) + { + while (!stoppingToken.IsCancellationRequested) + { + foreach (var uri in subscriptions) + { + await server.SendNotificationAsync("notifications/resource/updated", + new + { + Uri = uri, + }, cancellationToken: stoppingToken); + } + + await Task.Delay(5000, stoppingToken); + } + } +} diff --git a/samples/EverythingServer/Tools/AddTool.cs b/samples/EverythingServer/Tools/AddTool.cs new file mode 100644 index 00000000..ccaa306d --- /dev/null +++ b/samples/EverythingServer/Tools/AddTool.cs @@ -0,0 +1,11 @@ +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Tools; + +[McpServerToolType] +public class AddTool +{ + [McpServerTool(Name = "add"), Description("Adds two numbers.")] + public static string Add(int a, int b) => $"The sum of {a} and {b} is {a + b}"; +} diff --git a/samples/EverythingServer/Tools/AnnotatedMessageTool.cs b/samples/EverythingServer/Tools/AnnotatedMessageTool.cs new file mode 100644 index 00000000..25027faf --- /dev/null +++ b/samples/EverythingServer/Tools/AnnotatedMessageTool.cs @@ -0,0 +1,56 @@ +using ModelContextProtocol.Protocol.Types; +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Tools; + +[McpServerToolType] +public class AnnotatedMessageTool +{ + public enum MessageType + { + Error, + Success, + Debug, + } + + [McpServerTool(Name = "annotatedMessage"), Description("Generates an annotated message")] + public static IEnumerable AnnotatedMessage(MessageType messageType, bool includeImage = true) + { + List contents = messageType switch + { + MessageType.Error => [new() + { + Type = "text", + Text = "Error: Operation failed", + Annotations = new() { Audience = [Role.User, Role.Assistant], Priority = 1.0f } + }], + MessageType.Success => [new() + { + Type = "text", + Text = "Operation completed successfully", + Annotations = new() { Audience = [Role.User], Priority = 0.7f } + }], + MessageType.Debug => [new() + { + Type = "text", + Text = "Debug: Cache hit ratio 0.95, latency 150ms", + Annotations = new() { Audience = [Role.Assistant], Priority = 0.3f } + }], + _ => throw new ArgumentOutOfRangeException(nameof(messageType), messageType, null) + }; + + if (includeImage) + { + contents.Add(new() + { + Type = "image", + Data = TinyImageTool.MCP_TINY_IMAGE.Split(",").Last(), + MimeType = "image/png", + Annotations = new() { Audience = [Role.User], Priority = 0.5f } + }); + } + + return contents; + } +} diff --git a/samples/EverythingServer/Tools/EchoTool.cs b/samples/EverythingServer/Tools/EchoTool.cs new file mode 100644 index 00000000..6abd6d36 --- /dev/null +++ b/samples/EverythingServer/Tools/EchoTool.cs @@ -0,0 +1,11 @@ +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Tools; + +[McpServerToolType] +public class EchoTool +{ + [McpServerTool(Name = "echo"), Description("Echoes the message back to the client.")] + public static string Echo(string message) => $"Echo: {message}"; +} diff --git a/samples/EverythingServer/Tools/LongRunningTool.cs b/samples/EverythingServer/Tools/LongRunningTool.cs new file mode 100644 index 00000000..86acc84d --- /dev/null +++ b/samples/EverythingServer/Tools/LongRunningTool.cs @@ -0,0 +1,39 @@ +using ModelContextProtocol; +using ModelContextProtocol.Protocol.Messages; +using ModelContextProtocol.Protocol.Types; +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Tools; + +[McpServerToolType] +public class LongRunningTool +{ + [McpServerTool(Name = "longRunningOperation"), Description("Demonstrates a long running operation with progress updates")] + public static async Task LongRunningOperation( + IMcpServer server, + RequestContext context, + int duration = 10, + int steps = 5) + { + var progressToken = context.Params?.Meta?.ProgressToken; + var stepDuration = duration / steps; + + for (int i = 1; i <= steps + 1; i++) + { + await Task.Delay(stepDuration * 1000); + + if (progressToken is not null) + { + await server.SendNotificationAsync("notifications/progress", new + { + Progress = i, + Total = steps, + progressToken + }); + } + } + + return $"Long running operation completed. Duration: {duration} seconds. Steps: {steps}."; + } +} diff --git a/samples/EverythingServer/Tools/PrintEnvTool.cs b/samples/EverythingServer/Tools/PrintEnvTool.cs new file mode 100644 index 00000000..ca289b5f --- /dev/null +++ b/samples/EverythingServer/Tools/PrintEnvTool.cs @@ -0,0 +1,18 @@ +using ModelContextProtocol.Server; +using System.ComponentModel; +using System.Text.Json; + +namespace EverythingServer.Tools; + +[McpServerToolType] +public class PrintEnvTool +{ + private static readonly JsonSerializerOptions options = new() + { + WriteIndented = true + }; + + [McpServerTool(Name = "printEnv"), Description("Prints all environment variables, helpful for debugging MCP server configuration")] + public static string PrintEnv() => + JsonSerializer.Serialize(Environment.GetEnvironmentVariables(), options); +} diff --git a/samples/EverythingServer/Tools/SampleLlmTool.cs b/samples/EverythingServer/Tools/SampleLlmTool.cs new file mode 100644 index 00000000..53ebfff2 --- /dev/null +++ b/samples/EverythingServer/Tools/SampleLlmTool.cs @@ -0,0 +1,42 @@ +using ModelContextProtocol.Protocol.Types; +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Tools; + +[McpServerToolType] +public class SampleLlmTool +{ + [McpServerTool(Name = "sampleLLM"), Description("Samples from an LLM using MCP's sampling feature")] + public static async Task SampleLLM( + IMcpServer server, + [Description("The prompt to send to the LLM")] string prompt, + [Description("Maximum number of tokens to generate")] int maxTokens, + CancellationToken cancellationToken) + { + var samplingParams = CreateRequestSamplingParams(prompt ?? string.Empty, "sampleLLM", maxTokens); + var sampleResult = await server.RequestSamplingAsync(samplingParams, cancellationToken); + + return $"LLM sampling result: {sampleResult.Content.Text}"; + } + + private static CreateMessageRequestParams CreateRequestSamplingParams(string context, string uri, int maxTokens = 100) + { + return new CreateMessageRequestParams() + { + Messages = [new SamplingMessage() + { + Role = Role.User, + Content = new Content() + { + Type = "text", + Text = $"Resource {uri} context: {context}" + } + }], + SystemPrompt = "You are a helpful test server.", + MaxTokens = maxTokens, + Temperature = 0.7f, + IncludeContext = ContextInclusion.ThisServer + }; + } +} diff --git a/samples/EverythingServer/Tools/TinyImageTool.cs b/samples/EverythingServer/Tools/TinyImageTool.cs new file mode 100644 index 00000000..bd88ce98 --- /dev/null +++ b/samples/EverythingServer/Tools/TinyImageTool.cs @@ -0,0 +1,19 @@ +using Microsoft.Extensions.AI; +using ModelContextProtocol.Server; +using System.ComponentModel; + +namespace EverythingServer.Tools; + +[McpServerToolType] +public class TinyImageTool +{ + [McpServerTool(Name = "getTinyImage"), Description("Get a tiny image from the server")] + public static IEnumerable GetTinyImage() => [ + new TextContent("This is a tiny image:"), + new DataContent(MCP_TINY_IMAGE), + new TextContent("The image above is the MCP tiny image.") + ]; + + internal const string MCP_TINY_IMAGE = + ""; +} diff --git a/src/ModelContextProtocol/Configuration/McpServerBuilderExtensions.cs b/src/ModelContextProtocol/Configuration/McpServerBuilderExtensions.cs index a59de8ce..a564fd39 100644 --- a/src/ModelContextProtocol/Configuration/McpServerBuilderExtensions.cs +++ b/src/ModelContextProtocol/Configuration/McpServerBuilderExtensions.cs @@ -314,7 +314,7 @@ public static IMcpServerBuilder WithSubscribeToResourcesHandler(this IMcpServerB } /// - /// Sets or sets the handler for subscribe to resources messages. + /// Sets the handler for subscribe to resources messages. /// /// The builder instance. /// The handler. @@ -325,6 +325,18 @@ public static IMcpServerBuilder WithUnsubscribeFromResourcesHandler(this IMcpSer builder.Services.Configure(s => s.UnsubscribeFromResourcesHandler = handler); return builder; } + + /// + /// Sets the handler for setting the logging level. + /// + /// The builder instance. + /// The handler. + public static IMcpServerBuilder WithSetLoggingLevelHandler(this IMcpServerBuilder builder, Func, CancellationToken, Task> handler) + { + Throw.IfNull(builder); + builder.Services.Configure(s => s.SetLoggingLevelHandler = handler); + return builder; + } #endregion #region Transports diff --git a/src/ModelContextProtocol/Server/McpServerHandlers.cs b/src/ModelContextProtocol/Server/McpServerHandlers.cs index e472a7ae..b591182d 100644 --- a/src/ModelContextProtocol/Server/McpServerHandlers.cs +++ b/src/ModelContextProtocol/Server/McpServerHandlers.cs @@ -113,6 +113,7 @@ internal void OverwriteWithSetHandlers(McpServerOptions options) options.Capabilities.Prompts = promptsCapability; options.Capabilities.Resources = resourcesCapability; options.Capabilities.Tools = toolsCapability; + options.Capabilities.Logging = loggingCapability; options.GetCompletionHandler = GetCompletionHandler ?? options.GetCompletionHandler; } diff --git a/tests/ModelContextProtocol.Tests/Server/McpServerLoggingLevelTests.cs b/tests/ModelContextProtocol.Tests/Server/McpServerLoggingLevelTests.cs new file mode 100644 index 00000000..a97fe6f3 --- /dev/null +++ b/tests/ModelContextProtocol.Tests/Server/McpServerLoggingLevelTests.cs @@ -0,0 +1,55 @@ +using Microsoft.Extensions.DependencyInjection; +using ModelContextProtocol.Protocol.Types; +using ModelContextProtocol.Server; + +namespace ModelContextProtocol.Tests.Server; +public class McpServerLoggingLevelTests +{ + [Fact] + public void CanCreateServerWithLoggingLevelHandler() + { + var services = new ServiceCollection(); + + services.AddMcpServer() + .WithStdioServerTransport() + .WithSetLoggingLevelHandler((ctx, ct) => + { + return Task.FromResult(new EmptyResult()); + }); + + var provider = services.BuildServiceProvider(); + + provider.GetRequiredService(); + } + + [Fact] + public void AddingLoggingLevelHandlerSetsLoggingCapability() + { + var services = new ServiceCollection(); + + services.AddMcpServer() + .WithStdioServerTransport() + .WithSetLoggingLevelHandler((ctx, ct) => + { + return Task.FromResult(new EmptyResult()); + }); + + var provider = services.BuildServiceProvider(); + + var server = provider.GetRequiredService(); + + Assert.NotNull(server.ServerOptions.Capabilities?.Logging); + Assert.NotNull(server.ServerOptions.Capabilities.Logging.SetLoggingLevelHandler); + } + + [Fact] + public void ServerWithoutCallingLoggingLevelHandlerDoesNotSetLoggingCapability() + { + var services = new ServiceCollection(); + services.AddMcpServer() + .WithStdioServerTransport(); + var provider = services.BuildServiceProvider(); + var server = provider.GetRequiredService(); + Assert.Null(server.ServerOptions.Capabilities?.Logging); + } +}