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:

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:

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 stdiostateless 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 TypeJSON Schema Type
stringstring
int, longinteger
float, doublenumber
boolboolean
Complex typesobject 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-28 and 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).