mirror of
https://github.com/github/awesome-copilot.git
synced 2026-05-15 19:21:45 +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>
177 lines
6.5 KiB
Markdown
177 lines
6.5 KiB
Markdown
# Elicitation
|
|
|
|
Elicitation lets a tool **ask the user for input mid-execution**, via the client. The LLM doesn't see the question; the client surfaces it directly to the user. This turns one-shot tool calls into interactive flows — collecting confirmation, missing parameters, credentials (URL mode), etc.
|
|
|
|
> **Spec version:** 2025-11-25. URL mode is the newer addition (originally 2025-06-18 had only form mode).
|
|
|
|
## Two modes
|
|
|
|
| Mode | What it does | When to use |
|
|
|---|---|---|
|
|
| **Form (in-band)** | Server sends a JSON Schema; client renders a form; user submits values back through the same MCP channel. | Confirmations, missing parameters, structured choices. |
|
|
| **URL (out-of-band)** | Server sends a URL; client opens it in a browser; user completes the flow there; server checks state separately. | OAuth, payments, anything the MCP channel must not see. |
|
|
|
|
## Prerequisite: stateful transport
|
|
|
|
Elicitation requires the server to send a request *to* the client and wait for a response. That only works on:
|
|
- STDIO (always).
|
|
- Stateful HTTP (`options.Stateless = false`).
|
|
|
|
In stateless HTTP, `ElicitAsync` will throw — there's no transport channel back.
|
|
|
|
## Form mode — full example
|
|
|
|
```csharp
|
|
using System.ComponentModel;
|
|
using ModelContextProtocol.Protocol;
|
|
using ModelContextProtocol.Server;
|
|
|
|
[McpServerToolType]
|
|
public class BookingTools
|
|
{
|
|
[McpServerTool, Description("Books a meeting room. Asks the user for confirmation.")]
|
|
public static async Task<string> BookRoom(
|
|
IMcpServer server,
|
|
[Description("Room name")] string room,
|
|
[Description("Start time (ISO 8601)")] DateTime start,
|
|
CancellationToken ct)
|
|
{
|
|
var elicit = await server.ElicitAsync(new ElicitRequestParams
|
|
{
|
|
Message = $"Confirm booking '{room}' at {start:HH:mm}?",
|
|
RequestedSchema = new ElicitRequestParams.RequestSchema
|
|
{
|
|
Properties = new Dictionary<string, ElicitRequestParams.PrimitiveSchemaDefinition>
|
|
{
|
|
["confirm"] = new ElicitRequestParams.BooleanSchema
|
|
{
|
|
Description = "Confirm the booking",
|
|
Default = true
|
|
},
|
|
["notes"] = new ElicitRequestParams.StringSchema
|
|
{
|
|
Description = "Optional notes for the booking"
|
|
}
|
|
}
|
|
}
|
|
}, ct);
|
|
|
|
if (elicit.Action != "accept")
|
|
return "Booking cancelled by user.";
|
|
|
|
var confirmed = elicit.Content?["confirm"].GetBoolean() ?? false;
|
|
var notes = elicit.Content?["notes"].GetString() ?? "";
|
|
|
|
if (!confirmed)
|
|
return "User declined to confirm.";
|
|
|
|
// …perform the booking…
|
|
return $"Booked '{room}' at {start:O}. Notes: {notes}";
|
|
}
|
|
}
|
|
```
|
|
|
|
### Schema primitive types
|
|
|
|
You can build a `RequestedSchema` from these:
|
|
|
|
| Type | C# class | Notes |
|
|
|---|---|---|
|
|
| String | `StringSchema` | `Default`, `Description`. Add JSON-Schema validation at server side if you need it. |
|
|
| Number | `NumberSchema` | Use for ints and floats. |
|
|
| Boolean | `BooleanSchema` | Renders as a checkbox / toggle. |
|
|
| Single-select enum (untitled) | `UntitledSingleSelectEnumSchema` | List of values; client renders as dropdown/radio. |
|
|
| Single-select enum (titled) | `TitledSingleSelectEnumSchema` | Each value has a display title. |
|
|
| Multi-select enum | `UntitledMultiSelectEnumSchema` / `TitledMultiSelectEnumSchema` | Multi-select dropdown / checkbox group. |
|
|
|
|
Each accepts `Description` and `Default`.
|
|
|
|
### Response shape
|
|
|
|
`ElicitResult`:
|
|
- `Action` — `"accept"`, `"reject"`, or `"cancel"`. Always check this first.
|
|
- `Content` — `Dictionary<string, JsonElement>?` with the user's submitted values. `null` if the user rejected/cancelled.
|
|
|
|
Always handle the non-accept paths:
|
|
|
|
```csharp
|
|
if (elicit.Action == "cancel")
|
|
return "User cancelled. No changes made.";
|
|
if (elicit.Action == "reject")
|
|
return "User declined.";
|
|
// Action == "accept" → safe to read elicit.Content
|
|
```
|
|
|
|
## URL mode — full example
|
|
|
|
URL mode is for flows where the user must complete something **outside** the MCP channel — typically OAuth.
|
|
|
|
```csharp
|
|
[McpServerTool, Description("Connects the user's GitHub account.")]
|
|
public static async Task<string> ConnectGitHub(
|
|
IMcpServer server,
|
|
IOAuthService oauth,
|
|
CancellationToken ct)
|
|
{
|
|
var elicitationId = Guid.NewGuid().ToString();
|
|
var authUrl = oauth.BuildAuthorizationUrl(state: elicitationId);
|
|
|
|
var result = await server.ElicitAsync(new ElicitRequestParams
|
|
{
|
|
Mode = "url",
|
|
ElicitationId = elicitationId,
|
|
Url = authUrl,
|
|
Message = "Please authorize access to GitHub in the browser window that just opened."
|
|
}, ct);
|
|
|
|
if (result.Action != "accept")
|
|
return "Authorization cancelled.";
|
|
|
|
// The user has come back. Look up the persisted token by elicitationId.
|
|
var token = await oauth.GetTokenByStateAsync(elicitationId, ct);
|
|
return token is not null ? "Connected." : "Authorization did not complete.";
|
|
}
|
|
```
|
|
|
|
### `UrlElicitationRequiredException`
|
|
|
|
When a tool is *blocked* on auth (rather than walking the user through it), throw `UrlElicitationRequiredException`. The client surfaces the URL to the user and the call fails cleanly. Useful for retry-after-auth patterns:
|
|
|
|
```csharp
|
|
if (!oauth.HasValidToken)
|
|
{
|
|
var id = Guid.NewGuid().ToString();
|
|
throw new UrlElicitationRequiredException(
|
|
"Authorization required",
|
|
new[]
|
|
{
|
|
new ElicitRequestParams
|
|
{
|
|
Mode = "url",
|
|
ElicitationId = id,
|
|
Url = oauth.BuildAuthorizationUrl(state: id),
|
|
Message = "Sign in to continue."
|
|
}
|
|
});
|
|
}
|
|
```
|
|
|
|
## When NOT to use elicitation
|
|
|
|
- **Trivial confirmations the LLM can ask in natural language.** If you can phrase "Should I do X?" in your tool's docstring and let the LLM ask, that's lower friction than a modal form.
|
|
- **Branching that the LLM should reason about.** Don't replace the LLM's judgment with a form — only elicit for things the LLM literally cannot decide (user secrets, real-time consent, picking from a list only the user knows).
|
|
- **Stateless deployments.** Doesn't work — see prerequisite above.
|
|
|
|
## Client capability check
|
|
|
|
Don't blindly call `ElicitAsync`. Check first:
|
|
|
|
```csharp
|
|
if (server.ClientCapabilities?.Elicitation is null)
|
|
return "This client doesn't support elicitation; please pass the value as an argument.";
|
|
|
|
var elicit = await server.ElicitAsync(...);
|
|
```
|
|
|
|
This degrades gracefully on older clients.
|