Skip to content

Commit 01410df

Browse files
authored
Add EverythingServer sample (#151)
1 parent d21d933 commit 01410df

18 files changed

+620
-1
lines changed

ModelContextProtocol.sln

+7
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "QuickstartWeatherServer", "
5050
EndProject
5151
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "QuickstartClient", "samples\QuickstartClient\QuickstartClient.csproj", "{0D1552DC-E6ED-4AAC-5562-12F8352F46AA}"
5252
EndProject
53+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EverythingServer", "samples\EverythingServer\EverythingServer.csproj", "{17B8453F-AB72-99C5-E5EA-D0B065A6AE65}"
54+
EndProject
5355
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ModelContextProtocol.AspNetCore", "src\ModelContextProtocol.AspNetCore\ModelContextProtocol.AspNetCore.csproj", "{37B6A5E0-9995-497D-8B43-3BC6870CC716}"
5456
EndProject
5557
Global
@@ -94,6 +96,10 @@ Global
9496
{0D1552DC-E6ED-4AAC-5562-12F8352F46AA}.Debug|Any CPU.Build.0 = Debug|Any CPU
9597
{0D1552DC-E6ED-4AAC-5562-12F8352F46AA}.Release|Any CPU.ActiveCfg = Release|Any CPU
9698
{0D1552DC-E6ED-4AAC-5562-12F8352F46AA}.Release|Any CPU.Build.0 = Release|Any CPU
99+
{17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
100+
{17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Debug|Any CPU.Build.0 = Debug|Any CPU
101+
{17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Release|Any CPU.ActiveCfg = Release|Any CPU
102+
{17B8453F-AB72-99C5-E5EA-D0B065A6AE65}.Release|Any CPU.Build.0 = Release|Any CPU
97103
{37B6A5E0-9995-497D-8B43-3BC6870CC716}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
98104
{37B6A5E0-9995-497D-8B43-3BC6870CC716}.Debug|Any CPU.Build.0 = Debug|Any CPU
99105
{37B6A5E0-9995-497D-8B43-3BC6870CC716}.Release|Any CPU.ActiveCfg = Release|Any CPU
@@ -113,6 +119,7 @@ Global
113119
{0C6D0512-D26D-63D3-5019-C5F7A657B28C} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
114120
{4653EB0C-8FC0-98F4-E9C8-220EDA7A69DF} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
115121
{0D1552DC-E6ED-4AAC-5562-12F8352F46AA} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
122+
{17B8453F-AB72-99C5-E5EA-D0B065A6AE65} = {02EA681E-C7D8-13C7-8484-4AC65E1B71E8}
116123
{37B6A5E0-9995-497D-8B43-3BC6870CC716} = {A2F1F52A-9107-4BF8-8C3F-2F6670E7D0AD}
117124
EndGlobalSection
118125
GlobalSection(ExtensibilityGlobals) = postSolution
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
<Project Sdk="Microsoft.NET.Sdk">
2+
3+
<PropertyGroup>
4+
<TargetFramework>net9.0</TargetFramework>
5+
<Nullable>enable</Nullable>
6+
<ImplicitUsings>enable</ImplicitUsings>
7+
<OutputType>Exe</OutputType>
8+
</PropertyGroup>
9+
10+
<ItemGroup>
11+
<PackageReference Include="Microsoft.Extensions.Hosting" />
12+
</ItemGroup>
13+
14+
<ItemGroup>
15+
<ProjectReference Include="..\..\src\ModelContextProtocol\ModelContextProtocol.csproj" />
16+
</ItemGroup>
17+
18+
</Project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
using Microsoft.Extensions.DependencyInjection;
2+
using Microsoft.Extensions.Hosting;
3+
using ModelContextProtocol;
4+
using ModelContextProtocol.Protocol.Types;
5+
using ModelContextProtocol.Server;
6+
7+
namespace EverythingServer;
8+
9+
public class LoggingUpdateMessageSender(IMcpServer server, Func<LoggingLevel> getMinLevel) : BackgroundService
10+
{
11+
readonly Dictionary<LoggingLevel, string> _loggingLevelMap = new()
12+
{
13+
{ LoggingLevel.Debug, "Debug-level message" },
14+
{ LoggingLevel.Info, "Info-level message" },
15+
{ LoggingLevel.Notice, "Notice-level message" },
16+
{ LoggingLevel.Warning, "Warning-level message" },
17+
{ LoggingLevel.Error, "Error-level message" },
18+
{ LoggingLevel.Critical, "Critical-level message" },
19+
{ LoggingLevel.Alert, "Alert-level message" },
20+
{ LoggingLevel.Emergency, "Emergency-level message" }
21+
};
22+
23+
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
24+
{
25+
while (!stoppingToken.IsCancellationRequested)
26+
{
27+
var newLevel = (LoggingLevel)Random.Shared.Next(_loggingLevelMap.Count);
28+
29+
var message = new
30+
{
31+
Level = newLevel.ToString().ToLower(),
32+
Data = _loggingLevelMap[newLevel],
33+
};
34+
35+
if (newLevel > getMinLevel())
36+
{
37+
await server.SendNotificationAsync("notifications/message", message, cancellationToken: stoppingToken);
38+
}
39+
40+
await Task.Delay(15000, stoppingToken);
41+
}
42+
}
43+
}

samples/EverythingServer/Program.cs

+194
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,194 @@
1+
using EverythingServer;
2+
using EverythingServer.Prompts;
3+
using EverythingServer.Tools;
4+
using Microsoft.Extensions.AI;
5+
using Microsoft.Extensions.DependencyInjection;
6+
using Microsoft.Extensions.Hosting;
7+
using Microsoft.Extensions.Logging;
8+
using ModelContextProtocol;
9+
using ModelContextProtocol.Protocol.Types;
10+
using ModelContextProtocol.Server;
11+
12+
var builder = Host.CreateApplicationBuilder(args);
13+
builder.Logging.AddConsole(consoleLogOptions =>
14+
{
15+
// Configure all logs to go to stderr
16+
consoleLogOptions.LogToStandardErrorThreshold = LogLevel.Trace;
17+
});
18+
19+
HashSet<string> subscriptions = [];
20+
var _minimumLoggingLevel = LoggingLevel.Debug;
21+
22+
builder.Services
23+
.AddMcpServer()
24+
.WithStdioServerTransport()
25+
.WithTools<AddTool>()
26+
.WithTools<AnnotatedMessageTool>()
27+
.WithTools<EchoTool>()
28+
.WithTools<LongRunningTool>()
29+
.WithTools<PrintEnvTool>()
30+
.WithTools<SampleLlmTool>()
31+
.WithTools<TinyImageTool>()
32+
.WithPrompts<ComplexPromptType>()
33+
.WithPrompts<SimplePromptType>()
34+
.WithListResourceTemplatesHandler((ctx, ct) =>
35+
{
36+
return Task.FromResult(new ListResourceTemplatesResult
37+
{
38+
ResourceTemplates =
39+
[
40+
new ResourceTemplate { Name = "Static Resource", Description = "A static resource with a numeric ID", UriTemplate = "test://static/resource/{id}" }
41+
]
42+
});
43+
})
44+
.WithReadResourceHandler((ctx, ct) =>
45+
{
46+
var uri = ctx.Params?.Uri;
47+
48+
if (uri is null || !uri.StartsWith("test://static/resource/"))
49+
{
50+
throw new NotSupportedException($"Unknown resource: {uri}");
51+
}
52+
53+
int index = int.Parse(uri["test://static/resource/".Length..]) - 1;
54+
55+
if (index < 0 || index >= ResourceGenerator.Resources.Count)
56+
{
57+
throw new NotSupportedException($"Unknown resource: {uri}");
58+
}
59+
60+
var resource = ResourceGenerator.Resources[index];
61+
62+
if (resource.MimeType == "text/plain")
63+
{
64+
return Task.FromResult(new ReadResourceResult
65+
{
66+
Contents = [new TextResourceContents
67+
{
68+
Text = resource.Description!,
69+
MimeType = resource.MimeType,
70+
Uri = resource.Uri,
71+
}]
72+
});
73+
}
74+
else
75+
{
76+
return Task.FromResult(new ReadResourceResult
77+
{
78+
Contents = [new BlobResourceContents
79+
{
80+
Blob = resource.Description!,
81+
MimeType = resource.MimeType,
82+
Uri = resource.Uri,
83+
}]
84+
});
85+
}
86+
})
87+
.WithSubscribeToResourcesHandler(async (ctx, ct) =>
88+
{
89+
var uri = ctx.Params?.Uri;
90+
91+
if (uri is not null)
92+
{
93+
subscriptions.Add(uri);
94+
95+
await ctx.Server.RequestSamplingAsync([
96+
new ChatMessage(ChatRole.System, "You are a helpful test server"),
97+
new ChatMessage(ChatRole.User, $"Resource {uri}, context: A new subscription was started"),
98+
],
99+
options: new ChatOptions
100+
{
101+
MaxOutputTokens = 100,
102+
Temperature = 0.7f,
103+
},
104+
cancellationToken: ct);
105+
}
106+
107+
return new EmptyResult();
108+
})
109+
.WithUnsubscribeFromResourcesHandler((ctx, ct) =>
110+
{
111+
var uri = ctx.Params?.Uri;
112+
if (uri is not null)
113+
{
114+
subscriptions.Remove(uri);
115+
}
116+
return Task.FromResult(new EmptyResult());
117+
})
118+
.WithGetCompletionHandler((ctx, ct) =>
119+
{
120+
var exampleCompletions = new Dictionary<string, IEnumerable<string>>
121+
{
122+
{ "style", ["casual", "formal", "technical", "friendly"] },
123+
{ "temperature", ["0", "0.5", "0.7", "1.0"] },
124+
{ "resourceId", ["1", "2", "3", "4", "5"] }
125+
};
126+
127+
if (ctx.Params is not { } @params)
128+
{
129+
throw new NotSupportedException($"Params are required.");
130+
}
131+
132+
var @ref = @params.Ref;
133+
var argument = @params.Argument;
134+
135+
if (@ref.Type == "ref/resource")
136+
{
137+
var resourceId = @ref.Uri?.Split("/").Last();
138+
139+
if (resourceId is null)
140+
{
141+
return Task.FromResult(new CompleteResult());
142+
}
143+
144+
var values = exampleCompletions["resourceId"].Where(id => id.StartsWith(argument.Value));
145+
146+
return Task.FromResult(new CompleteResult
147+
{
148+
Completion = new Completion { Values = [..values], HasMore = false, Total = values.Count() }
149+
});
150+
}
151+
152+
if (@ref.Type == "ref/prompt")
153+
{
154+
if (!exampleCompletions.TryGetValue(argument.Name, out IEnumerable<string>? value))
155+
{
156+
throw new NotSupportedException($"Unknown argument name: {argument.Name}");
157+
}
158+
159+
var values = value.Where(value => value.StartsWith(argument.Value));
160+
return Task.FromResult(new CompleteResult
161+
{
162+
Completion = new Completion { Values = [..values], HasMore = false, Total = values.Count() }
163+
});
164+
}
165+
166+
throw new NotSupportedException($"Unknown reference type: {@ref.Type}");
167+
})
168+
.WithSetLoggingLevelHandler(async (ctx, ct) =>
169+
{
170+
if (ctx.Params?.Level is null)
171+
{
172+
throw new McpException("Missing required argument 'level'");
173+
}
174+
175+
_minimumLoggingLevel = ctx.Params.Level;
176+
177+
await ctx.Server.SendNotificationAsync("notifications/message", new
178+
{
179+
Level = "debug",
180+
Logger = "test-server",
181+
Data = $"Logging level set to {_minimumLoggingLevel}",
182+
}, cancellationToken: ct);
183+
184+
return new EmptyResult();
185+
})
186+
;
187+
188+
builder.Services.AddSingleton(subscriptions);
189+
builder.Services.AddHostedService<SubscriptionMessageSender>();
190+
builder.Services.AddHostedService<LoggingUpdateMessageSender>();
191+
192+
builder.Services.AddSingleton<Func<LoggingLevel>>(_ => () => _minimumLoggingLevel);
193+
194+
await builder.Build().RunAsync();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
using EverythingServer.Tools;
2+
using Microsoft.Extensions.AI;
3+
using ModelContextProtocol.Server;
4+
using System.ComponentModel;
5+
6+
namespace EverythingServer.Prompts;
7+
8+
[McpServerPromptType]
9+
public class ComplexPromptType
10+
{
11+
[McpServerPrompt(Name = "complex_prompt"), Description("A prompt with arguments")]
12+
public static IEnumerable<ChatMessage> ComplexPrompt(
13+
[Description("Temperature setting")] int temperature,
14+
[Description("Output style")] string? style = null)
15+
{
16+
return [
17+
new ChatMessage(ChatRole.User,$"This is a complex prompt with arguments: temperature={temperature}, style={style}"),
18+
new ChatMessage(ChatRole.Assistant, "I understand. You've provided a complex prompt with temperature and style arguments. How would you like me to proceed?"),
19+
new ChatMessage(ChatRole.User, [new DataContent(TinyImageTool.MCP_TINY_IMAGE)])
20+
];
21+
}
22+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
using ModelContextProtocol.Server;
2+
using System.ComponentModel;
3+
4+
namespace EverythingServer.Prompts;
5+
6+
[McpServerPromptType]
7+
public class SimplePromptType
8+
{
9+
[McpServerPrompt(Name = "simple_prompt"), Description("A prompt without arguments")]
10+
public static string SimplePrompt() => "This is a simple prompt without arguments";
11+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
using ModelContextProtocol.Protocol.Types;
2+
using System;
3+
using System.Collections.Generic;
4+
using System.Linq;
5+
6+
namespace EverythingServer;
7+
8+
static class ResourceGenerator
9+
{
10+
private static readonly List<Resource> _resources = Enumerable.Range(1, 100).Select(i =>
11+
{
12+
var uri = $"test://static/resource/{i}";
13+
if (i % 2 != 0)
14+
{
15+
return new Resource
16+
{
17+
Uri = uri,
18+
Name = $"Resource {i}",
19+
MimeType = "text/plain",
20+
Description = $"Resource {i}: This is a plaintext resource"
21+
};
22+
}
23+
else
24+
{
25+
var buffer = System.Text.Encoding.UTF8.GetBytes($"Resource {i}: This is a base64 blob");
26+
return new Resource
27+
{
28+
Uri = uri,
29+
Name = $"Resource {i}",
30+
MimeType = "application/octet-stream",
31+
Description = Convert.ToBase64String(buffer)
32+
};
33+
}
34+
}).ToList();
35+
36+
public static IReadOnlyList<Resource> Resources => _resources;
37+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
using Microsoft.Extensions.Hosting;
2+
using ModelContextProtocol;
3+
using ModelContextProtocol.Server;
4+
5+
internal class SubscriptionMessageSender(IMcpServer server, HashSet<string> subscriptions) : BackgroundService
6+
{
7+
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
8+
{
9+
while (!stoppingToken.IsCancellationRequested)
10+
{
11+
foreach (var uri in subscriptions)
12+
{
13+
await server.SendNotificationAsync("notifications/resource/updated",
14+
new
15+
{
16+
Uri = uri,
17+
}, cancellationToken: stoppingToken);
18+
}
19+
20+
await Task.Delay(5000, stoppingToken);
21+
}
22+
}
23+
}
+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
using ModelContextProtocol.Server;
2+
using System.ComponentModel;
3+
4+
namespace EverythingServer.Tools;
5+
6+
[McpServerToolType]
7+
public class AddTool
8+
{
9+
[McpServerTool(Name = "add"), Description("Adds two numbers.")]
10+
public static string Add(int a, int b) => $"The sum of {a} and {b} is {a + b}";
11+
}

0 commit comments

Comments
 (0)