mirror of
https://github.com/github/awesome-copilot.git
synced 2026-05-16 03:31:47 +00:00
2c275f2ef9
* feat(skills): add dotnet-mcp-builder, deprecate csharp-mcp-server-generator Adds a comprehensive skill for building MCP (Model Context Protocol) servers in C#/.NET against the official ModelContextProtocol 1.x NuGet packages. Covers both transports (STDIO, Streamable HTTP — SSE is deprecated) and every primitive in the current MCP spec (2025-11-25): tools, prompts, resources, elicitation (form + URL mode), sampling, roots, completions, logging, and MCP Apps. Includes a thin .NET MCP client reference and testing guidance (MCP Inspector + in-memory transport for unit tests). Steers the model toward the current stable 1.x packages instead of the 0.x previews it tends to pin by default, and enforces the STDIO stdout/stderr trap. Also deprecates the existing csharp-mcp-server-generator skill, which predates ModelContextProtocol 1.0 and only covered a subset of the current spec. Its SKILL.md now redirects users to dotnet-mcp-builder so existing install URLs keep working without surprises. * fix: address PR review from aaronpowell - Delete csharp-mcp-server-generator skill (rather than deprecating it) - Update mcp-apps.md pitfalls section to reference .NET Tool.Meta type instead of the serialized _meta JSON property names - Rebuild docs/README.skills.md Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com> * chore: remove C# MCP development plugin files * chore: remove csharp-mcp-development plugin entry from marketplace --------- Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
190 lines
6.8 KiB
Markdown
190 lines
6.8 KiB
Markdown
# Streamable HTTP transport (ASP.NET Core)
|
|
|
|
Streamable HTTP is the modern remote transport. A single endpoint accepts JSON-RPC over HTTP POST and (optionally) streams responses back as Server-Sent Events when the server has more than one message to send.
|
|
|
|
> **SSE-only is deprecated.** The legacy "HTTP+SSE" transport (separate POST endpoint + GET SSE endpoint) is gone from new clients. Use Streamable HTTP. Only enable legacy SSE (`EnableLegacySse = true`) if you must support a known-old client, and document why.
|
|
|
|
## When to choose HTTP
|
|
|
|
- Multi-tenant or remote-hosted server.
|
|
- Auth via OAuth / API gateway in front.
|
|
- Horizontally scaled deployments (with `Stateless = true`).
|
|
- Containers, Azure Container Apps, Kubernetes, etc.
|
|
|
|
For local single-user scenarios, [STDIO](./transport-stdio.md) is simpler.
|
|
|
|
## Minimal server
|
|
|
|
```bash
|
|
dotnet new web -n MyHttpServer -f net10.0
|
|
cd MyHttpServer
|
|
dotnet add package ModelContextProtocol.AspNetCore
|
|
```
|
|
|
|
```csharp
|
|
// Program.cs
|
|
using ModelContextProtocol.Server;
|
|
using System.ComponentModel;
|
|
|
|
var builder = WebApplication.CreateBuilder(args);
|
|
|
|
builder.Services
|
|
.AddMcpServer()
|
|
.WithHttpTransport(options =>
|
|
{
|
|
// Stateless = true: each request is independent, no Mcp-Session-Id tracking.
|
|
// Required for horizontal scaling without sticky sessions.
|
|
// Disables server-to-client features (sampling, elicitation, roots, unsolicited notifications).
|
|
options.Stateless = true;
|
|
})
|
|
.WithToolsFromAssembly();
|
|
|
|
var app = builder.Build();
|
|
|
|
app.MapMcp(); // mounts the MCP endpoints at "/"
|
|
// app.MapMcp("/mcp"); // or under a path prefix
|
|
|
|
app.Run("http://localhost:3001");
|
|
|
|
[McpServerToolType]
|
|
public static class EchoTool
|
|
{
|
|
[McpServerTool, Description("Echoes the message back to the client.")]
|
|
public static string Echo(string message) => $"hello {message}";
|
|
}
|
|
```
|
|
|
|
## Stateless vs. stateful — the most important decision
|
|
|
|
| Mode | `options.Stateless` | Behaviour | Use when |
|
|
|---|---|---|---|
|
|
| **Stateless** | `true` | No `Mcp-Session-Id`. Each POST is independent. | Horizontal scaling, simple tool servers, no server-initiated traffic. |
|
|
| **Stateful** | `false` (default) | Server assigns and tracks `Mcp-Session-Id`. Long-lived session. | You need elicitation, sampling, roots, log notifications, or anything that pushes from server to client. Requires session affinity at the load balancer. |
|
|
|
|
**Rule:** if the user wants any of `ElicitAsync`, `SampleAsync`, `RequestRootsAsync`, or to push log/notification messages, **do not** set `Stateless = true`. The calls will fail at runtime with no transport to deliver them on.
|
|
|
|
## Endpoint shape
|
|
|
|
`MapMcp(pattern = "")` creates a route group at `pattern` and maps:
|
|
- **POST** — accepts JSON-RPC requests/responses/notifications. Returns either a JSON response or an SSE stream depending on `Accept` header and whether multiple messages need to flow back.
|
|
- **GET** — used by stateful sessions for the server-to-client SSE channel.
|
|
- **DELETE** — terminates a stateful session.
|
|
|
|
Default pattern is the root (`/`). To put MCP under `/mcp/v1`:
|
|
|
|
```csharp
|
|
app.MapMcp("/mcp/v1");
|
|
```
|
|
|
|
Match this on the client side (`Endpoint = new Uri("https://host/mcp/v1")`).
|
|
|
|
## Per-session configuration (HttpContext access)
|
|
|
|
When you need to vary server behaviour per HTTP request (auth, tenant, headers), use the `ConfigureSessionOptions` callback:
|
|
|
|
```csharp
|
|
builder.Services
|
|
.AddMcpServer()
|
|
.WithHttpTransport(options =>
|
|
{
|
|
options.ConfigureSessionOptions = async (httpContext, mcpOptions, ct) =>
|
|
{
|
|
var tenantId = httpContext.Request.Headers["X-Tenant"].ToString();
|
|
mcpOptions.ServerInstructions = $"Tenant: {tenantId}";
|
|
// mutate any McpServerOptions fields per-session
|
|
};
|
|
});
|
|
```
|
|
|
|
Inside a tool, you can also inject `IHttpContextAccessor` if `AddHttpContextAccessor()` is registered. See the [`AspNetCoreMcpPerSessionTools` sample](https://github.com/modelcontextprotocol/csharp-sdk/tree/main/samples/AspNetCoreMcpPerSessionTools).
|
|
|
|
## Authentication
|
|
|
|
The MCP endpoint is just an ASP.NET Core endpoint — apply standard middleware:
|
|
|
|
```csharp
|
|
builder.Services
|
|
.AddAuthentication("Bearer")
|
|
.AddJwtBearer(/* configure */);
|
|
builder.Services.AddAuthorization();
|
|
|
|
var app = builder.Build();
|
|
|
|
app.UseAuthentication();
|
|
app.UseAuthorization();
|
|
|
|
app.MapMcp().RequireAuthorization(); // protect the endpoint
|
|
```
|
|
|
|
For OAuth flows where the *MCP server* is the resource server, follow the [MCP authorization spec](https://modelcontextprotocol.io/specification/2025-06-18/basic/authorization). The [`ProtectedMcpServer` sample](https://github.com/modelcontextprotocol/csharp-sdk/tree/main/samples/ProtectedMcpServer) shows a working setup with discovery endpoints.
|
|
|
|
For machine-to-machine, an API key middleware is fine:
|
|
|
|
```csharp
|
|
app.Use(async (ctx, next) =>
|
|
{
|
|
if (ctx.Request.Headers["X-Api-Key"] != Configuration["ApiKey"])
|
|
{
|
|
ctx.Response.StatusCode = StatusCodes.Status401Unauthorized;
|
|
return;
|
|
}
|
|
await next();
|
|
});
|
|
```
|
|
|
|
## CORS (when the client is in-browser)
|
|
|
|
```csharp
|
|
builder.Services.AddCors(o => o.AddDefaultPolicy(p =>
|
|
p.WithOrigins("https://my-host.example.com")
|
|
.AllowAnyHeader()
|
|
.AllowAnyMethod()
|
|
.AllowCredentials()));
|
|
// ...
|
|
app.UseCors();
|
|
app.MapMcp();
|
|
```
|
|
|
|
## Health checks and observability
|
|
|
|
Add the standard ASP.NET Core probes; the MCP endpoint shouldn't be the liveness check.
|
|
|
|
```csharp
|
|
builder.Services.AddHealthChecks();
|
|
// ...
|
|
app.MapHealthChecks("/healthz");
|
|
```
|
|
|
|
The SDK emits OpenTelemetry traces (`Activity` per tool call) and metrics. Wire them up if the user has an OTel pipeline:
|
|
|
|
```csharp
|
|
builder.Services
|
|
.AddOpenTelemetry()
|
|
.WithTracing(t => t.AddSource("ModelContextProtocol").AddOtlpExporter())
|
|
.WithMetrics(m => m.AddMeter("ModelContextProtocol").AddOtlpExporter());
|
|
```
|
|
|
|
## Deployment notes
|
|
|
|
- **Containerise normally.** No special MCP-specific Dockerfile — it's just an ASP.NET Core app.
|
|
- **Behind a reverse proxy** (nginx, Azure Front Door, AWS ALB), make sure SSE buffering is **disabled** for the MCP path. nginx: `proxy_buffering off;`. Without this, streaming responses are batched into one slow blob.
|
|
- **Timeouts.** The client may keep an SSE connection open for a long time. Set proxy idle timeout high (e.g. 5+ minutes) for stateful deployments; less critical for stateless.
|
|
- **Azure Container Apps / App Service** work out of the box; both support long-lived HTTP responses.
|
|
|
|
## Enabling legacy SSE (compatibility only)
|
|
|
|
```csharp
|
|
builder.Services
|
|
.AddMcpServer()
|
|
.WithHttpTransport(options =>
|
|
{
|
|
options.EnableLegacySse = true;
|
|
#pragma warning disable MCP9004
|
|
options.Stateless = false; // SSE requires stateful mode
|
|
#pragma warning restore MCP9004
|
|
})
|
|
.WithToolsFromAssembly();
|
|
```
|
|
|
|
Only do this if the user has a documented client that hasn't migrated. New deployments should not enable it.
|