tools.md
June 24, 2026 · View on GitHub
Tools
MCP tools allow servers to expose callable functions to clients. Tools are the primary mechanism for LLMs to take action through MCP—they enable everything from querying databases to calling web APIs.
This document covers tool content types, change notifications, and schema generation.
Defining tools on the server
Tools can be defined in several ways:
- Using the xref:ModelContextProtocol.Server.McpServerToolAttribute attribute on methods within a class marked with xref:ModelContextProtocol.Server.McpServerToolTypeAttribute
- Using xref:ModelContextProtocol.Server.McpServerTool.Create* factory methods from a delegate,
MethodInfo, orAIFunction - Deriving from xref:ModelContextProtocol.Server.McpServerTool or xref:ModelContextProtocol.Server.DelegatingMcpServerTool
- Implementing a custom xref:ModelContextProtocol.Server.McpRequestHandler`2 via xref:ModelContextProtocol.Server.McpServerHandlers
- Implementing a low-level xref:ModelContextProtocol.Server.McpRequestFilter`2
The attribute-based approach is the most common and is shown throughout this document. Parameters are automatically deserialized from JSON and documented using [Description] attributes. In addition to tool arguments, methods can accept special parameter types that are resolved automatically: xref:ModelContextProtocol.Server.McpServer, IProgress<ProgressNotificationValue>, ClaimsPrincipal, and any service registered through dependency injection.
[McpServerToolType]
public class MyTools
{
[McpServerTool, Description("Echoes the input message back")]
public static string Echo([Description("The message to echo")] string message)
=> $"Echo: {message}";
}
Register the tool type when building the server:
builder.Services.AddMcpServer()
.WithHttpTransport(o => o.Stateless = true)
.WithTools<MyTools>();
Content types
Tools can return various content types. The simplest is a string, which is automatically wrapped in a xref:ModelContextProtocol.Protocol.TextContentBlock. For richer content, tools can return one or more xref:ModelContextProtocol.Protocol.ContentBlock instances. Tools can also return DataContent from Microsoft.Extensions.AI, which is automatically mapped to the appropriate MCP content block: image MIME types become xref:ModelContextProtocol.Protocol.ImageContentBlock, audio MIME types become xref:ModelContextProtocol.Protocol.AudioContentBlock, and all other MIME types become xref:ModelContextProtocol.Protocol.EmbeddedResourceBlock with binary resource contents.
Text content
Return a string or a xref:ModelContextProtocol.Protocol.TextContentBlock directly:
[McpServerTool, Description("Returns a greeting")]
public static string Greet(string name) => $"Hello, {name}!";
Image content
Return an xref:ModelContextProtocol.Protocol.ImageContentBlock with base64-encoded image data and a MIME type. Use the xref:ModelContextProtocol.Protocol.ImageContentBlock.FromBytes* factory method or construct the block directly:
[McpServerTool, Description("Returns a generated image")]
public static ImageContentBlock GenerateImage()
{
byte[] pngBytes = CreateImage(); // your image generation logic
return ImageContentBlock.FromBytes(pngBytes, "image/png");
}
Audio content
Return an xref:ModelContextProtocol.Protocol.AudioContentBlock with base64-encoded audio data and a MIME type. The xref:ModelContextProtocol.Protocol.AudioContentBlock.FromBytes* factory method encodes the raw bytes automatically:
[McpServerTool, Description("Returns a synthesized audio clip")]
public static AudioContentBlock Synthesize(string text)
{
byte[] wavBytes = TextToSpeech(text); // your audio synthesis logic
return AudioContentBlock.FromBytes(wavBytes, "audio/wav");
}
Supported audio MIME types include audio/wav, audio/mp3, audio/ogg, and others depending on what the client can handle.
Embedded resources
Return an xref:ModelContextProtocol.Protocol.EmbeddedResourceBlock to embed a resource directly in a tool result. The resource can contain either text or binary data through xref:ModelContextProtocol.Protocol.TextResourceContents or xref:ModelContextProtocol.Protocol.BlobResourceContents:
[McpServerTool, Description("Returns a document as an embedded resource")]
public static EmbeddedResourceBlock GetDocument()
{
return new EmbeddedResourceBlock
{
Resource = new TextResourceContents
{
Uri = "docs://readme",
MimeType = "text/plain",
Text = "This is the document content."
}
};
}
For binary resources, use xref:ModelContextProtocol.Protocol.BlobResourceContents:
[McpServerTool, Description("Returns a binary resource")]
public static EmbeddedResourceBlock GetBinaryData(string id)
{
byte[] data = LoadData(id); // application logic to load data by ID
return new EmbeddedResourceBlock
{
Resource = BlobResourceContents.FromBytes(data, $"data://items/{id}", "application/octet-stream")
};
}
Mixed content
Tools can return multiple content blocks by returning IEnumerable<ContentBlock>:
[McpServerTool, Description("Returns text and an image")]
public static IEnumerable<ContentBlock> DescribeImage()
{
byte[] imageBytes = GetImage();
return
[
new TextContentBlock { Text = "Here is the generated image:" },
ImageContentBlock.FromBytes(imageBytes, "image/png"),
new TextContentBlock { Text = "The image shows a landscape." }
];
}
Content annotations
Any content block can include xref:ModelContextProtocol.Protocol.Annotations to provide hints about the intended audience and priority:
new TextContentBlock
{
Text = "Detailed debug information",
Annotations = new Annotations
{
Audience = [Role.Assistant], // Only for the LLM, not the user
Priority = 0.3f // Low priority (0.0 to 1.0)
}
}
Consuming tools on the client
Clients can discover and call tools using xref:ModelContextProtocol.Client.McpClient:
// List available tools
IList<McpClientTool> tools = await client.ListToolsAsync();
foreach (var tool in tools)
{
Console.WriteLine($"{tool.Name}: {tool.Description}");
}
// Call a tool by finding it in the list
McpClientTool echoTool = tools.First(t => t.Name == "echo");
CallToolResult result = await echoTool.CallAsync(
new Dictionary<string, object?> { ["message"] = "Hello!" });
// Process the result content blocks
foreach (var content in result.Content)
{
switch (content)
{
case TextContentBlock text:
Console.WriteLine(text.Text);
break;
case ImageContentBlock image:
File.WriteAllBytes("output.png", image.DecodedData.ToArray());
break;
case AudioContentBlock audio:
File.WriteAllBytes("output.wav", audio.DecodedData.ToArray());
break;
case EmbeddedResourceBlock resource:
if (resource.Resource is TextResourceContents textResource)
Console.WriteLine(textResource.Text);
break;
}
}
Error handling
Tool errors in MCP are distinct from protocol errors. When a tool encounters an error during execution, the error is reported inside the xref:ModelContextProtocol.Protocol.CallToolResult with xref:ModelContextProtocol.Protocol.CallToolResult.IsError set to true, rather than as a protocol-level exception. This allows the LLM to see the error and potentially recover.
Automatic exception handling
When a tool method throws an exception, the server catches it and returns a CallToolResult with IsError = true, with the following exceptions:
- xref:ModelContextProtocol.McpProtocolException is re-thrown as a JSON-RPC error response (not a tool error result).
OperationCanceledExceptionis re-thrown when the cancellation token was triggered.
For all other exceptions, the error is returned as a tool result. If the exception derives from xref:ModelContextProtocol.McpException (excluding McpProtocolException, which is re-thrown above), its message is included in the error text; otherwise, a generic message is returned to avoid leaking internal details.
[McpServerTool, Description("Divides two numbers")]
public static double Divide(double a, double b)
{
if (b == 0)
{
// ArgumentException is not an McpException, so the client receives a generic message:
// "An error occurred invoking 'divide'."
throw new ArgumentException("Cannot divide by zero");
}
return a / b;
}
Protocol errors
Throw xref:ModelContextProtocol.McpProtocolException to signal a protocol-level error (e.g., invalid parameters or unknown tool). These exceptions propagate as JSON-RPC error responses rather than tool error results:
[McpServerTool, Description("Processes the input")]
public static string Process(string input)
{
if (string.IsNullOrEmpty(input))
{
// Propagates as a JSON-RPC error with code -32602 (InvalidParams)
// and message "Missing required input"
throw new McpProtocolException("Missing required input", McpErrorCode.InvalidParams);
}
return $"Processed: {input}";
}
Checking for errors on the client
On the client side, inspect the xref:ModelContextProtocol.Protocol.CallToolResult.IsError property after calling a tool:
CallToolResult result = await client.CallToolAsync("divide", new Dictionary<string, object?>
{
["a"] = 10,
["b"] = 0
});
if (result.IsError is true)
{
// Prints: "Tool error: An error occurred invoking 'divide'."
Console.WriteLine($"Tool error: {result.Content.OfType<TextContentBlock>().FirstOrDefault()?.Text}");
}
Tool list change notifications
Servers can dynamically add, remove, or modify tools at runtime. When the tool list changes, the server notifies connected clients so they can refresh their tool list. These are unsolicited notifications, so they require stateful mode or stdio — stateless servers cannot send unsolicited notifications.
Sending notifications from the server
Inject xref:ModelContextProtocol.Server.McpServer and call the notification method after modifying the tool list:
// After adding or removing tools dynamically
await server.SendNotificationAsync(
NotificationMethods.ToolListChangedNotification,
new ToolListChangedNotificationParams());
Handling notifications on the client
Register a notification handler on the client to respond to tool list changes:
mcpClient.RegisterNotificationHandler(
NotificationMethods.ToolListChangedNotification,
async (notification, cancellationToken) =>
{
// Refresh the tool list
var updatedTools = await mcpClient.ListToolsAsync(cancellationToken: cancellationToken);
Console.WriteLine($"Tool list updated. {updatedTools.Count} tools available.");
});
JSON Schema generation
Tool parameters are described using JSON Schema 2020-12. JSON schemas are automatically generated from .NET method signatures when the [McpServerTool] attribute is applied. Parameter types are mapped to JSON Schema types:
| .NET Type | JSON Schema Type |
|---|---|
string | string |
int, long | integer |
float, double | number |
bool | boolean |
| Complex types | object with properties |
Use [Description] attributes on parameters to populate the description field in the generated schema. This helps LLMs understand what each parameter expects.
[McpServerTool, Description("Searches for items")]
public static string Search(
[Description("The search query string")] string query,
[Description("Maximum results to return (1-100)")] int maxResults = 10)
{
// Schema will include descriptions and default value for maxResults
}
Custom HTTP headers from tool parameters
When using the Streamable HTTP transport, tool parameters can be mirrored as HTTP headers so that network infrastructure (load balancers, proxies, gateways) can make routing decisions without parsing the JSON-RPC request body. Apply the xref:ModelContextProtocol.Server.McpHeaderAttribute to a parameter to opt it in:
[McpServerTool, Description("Executes a SQL query in a specific region")]
public static string ExecuteSql(
[McpHeader("Region"), Description("Target datacenter region")] string region,
[Description("The SQL query to execute")] string query)
{
// Clients will send an additional HTTP header:
// Mcp-Param-Region: <region value>
}
When the tool's schema is generated, the annotated parameter includes an x-mcp-header extension property. Clients read this annotation and automatically add the corresponding Mcp-Param-{Name} header on outgoing tools/call requests. The server validates that the header value matches the value in the JSON-RPC body.
Rules and constraints:
- Only primitive parameter types (
string, numeric types,bool) are supported. - The header name must contain only visible ASCII characters (0x21–0x7E) excluding colon (
:). - Values containing non-ASCII characters, control characters, or leading/trailing whitespace are Base64-encoded using the
=?base64?{value}?=wrapper. - Header names must be case-insensitively unique within the tool's input schema.
- Header validation is enforced only for protocol versions that support the HTTP Standardization feature (currently
2026-07-28and later).
Pre-loading tool definitions on the client
By default, Mcp-Param-* headers are sent only for tools discovered via xref:ModelContextProtocol.Client.McpClient.ListToolsAsync*. If a client already has tool schema information (for example, from a previous session, hardcoded configuration, or an out-of-band source), it can pre-load those definitions so that headers are sent immediately—without a round trip to the server.
// Build the tool definition with x-mcp-header annotations
var tool = new Tool
{
Name = "execute_sql",
InputSchema = JsonDocument.Parse("""
{
"type": "object",
"properties": {
"region": {
"type": "string",
"x-mcp-header": "Region"
},
"query": {
"type": "string"
}
}
}
""").RootElement.Clone(),
};
// Pre-load the tool definition — no ListToolsAsync needed
client.AddKnownTools([tool]);
// This call now sends an Mcp-Param-Region header automatically
var result = await client.CallToolAsync("execute_sql",
new Dictionary<string, object?> { ["region"] = "us-west-2", ["query"] = "SELECT 1" });
Known tools survive xref:ModelContextProtocol.Client.McpClient.ListToolsAsync* cache clears—they remain in the cache even when the server's tool list is refreshed. If the server returns a tool with the same name, the server's definition overwrites the cached one, but the tool keeps its known status.
To remove known tools, use xref:ModelContextProtocol.Client.McpClient.RemoveKnownTools* for specific tools or xref:ModelContextProtocol.Client.McpClient.ClearKnownTools* to remove all:
// Remove specific known tools by name
client.RemoveKnownTools(["execute_sql"]);
// Or remove all known tools at once
client.ClearKnownTools();
All tools passed to xref:ModelContextProtocol.Client.McpClient.AddKnownTools* are validated for correct x-mcp-header annotations. If any tool in the batch fails validation, an xref:System.ArgumentException is thrown and no tools are added (all-or-nothing).