docs/advanced/extensions.md
An extension is an opt-in bundle of MCP behaviour behind one identifier.
On a server it can contribute tools, resources, and new request methods, and it can wrap
tools/call. On a client it can claim extra tools/call result shapes and observe vendor
notifications. Each side advertises under its own capabilities.extensions, and nothing
changes for anyone who didn't ask for it. That is the contract (SEP-2133), and
it has one golden rule: extensions are off by default.
Pass instances at construction:
--8<-- "docs_src/extensions/tutorial001.py"
Done. The server now advertises io.modelcontextprotocol/ui under
capabilities.extensions and serves everything the extension contributes.
Apps is the built-in reference extension, and it gets its own page: MCP Apps.
!!! note
Extensions are fixed at construction. There is no add_extension to call later:
a server's capability map should not change while clients are connected to it.
The capability map rides server/discover, which is a 2026-07-28 path. A legacy
initialize handshake has nowhere to put it, so a legacy client simply doesn't see
the extension. Design for that: an extension augments a server, it must not be the
only way the server is usable.
Subclass Extension and override only what you need. Every method has a default.
--8<-- "docs_src/extensions/tutorial002.py"
The identifier is a vendor-prefix/name string following the spec's _meta key
grammar: dot-separated labels (each starts with a letter, ends with a letter or
digit), a slash, then the name. It is validated when the class is defined, so a
typo doesn't wait for a server to boot:
TypeError: Stamps.identifier must be a `vendor-prefix/name` string
(reverse-DNS prefix required), got 'stamps'
Use a domain you control as the prefix. io.modelcontextprotocol/* is for extensions
specified by the MCP project itself.
The smallest useful extension is one tool and a settings map:
--8<-- "docs_src/extensions/tutorial003.py"
tools() returns ToolBindings. The server registers each one exactly as if you
had called mcp.add_tool(...) yourself: same schema generation, same Context
injection, same everything.settings() is the value advertised at capabilities.extensions["com.example/stamps"].
Return {} (the default) to advertise the extension with no settings.MCPServer consumes them. There is no self.server to mutate.And main() is the proof, an in-memory client straight against mcp:
--8<-- "docs_src/extensions/tutorial003.py"
An extension can register new request methods: its own verbs, served next to the spec's:
--8<-- "docs_src/extensions/tutorial004.py"
SearchParams subclasses RequestParams, so the 2026 _meta envelope parses
uniformly and your handler gets validated params, never a raw dict. Bound what
the client controls: Field(ge=1, le=100) rejects an absurd limit before
your code allocates anything for it.require_client_extension(ctx, EXTENSION_ID) is the gate: a client that did not
declare the extension gets the -32021 (missing required client capability) error,
with the machine-readable requiredCapabilities payload the spec asks for.protocol_versions=frozenset({"2026-07-28"}) pins the method to one wire version.
At any other version the client gets METHOD_NOT_FOUND, exactly as if the method
didn't exist there. For that client, it doesn't.Methods are strictly additive. The SDK enforces this at construction, not at runtime:
MethodBinding for a spec-defined method (tools/list, completion/complete, ...)
raises ValueError when the binding is constructed. Core verbs belong to the server.protocol_versions set raises too: a method that can never be served
is a bug, not a configuration.The same file's main() is the whole client story, both halves of it:
--8<-- "docs_src/extensions/tutorial004.py"
Client(..., extensions=[advertise(EXTENSION_ID)]) declares the extension. The
declarations become ClientCapabilities.extensions: on a 2026-07-28 connection
the map travels in the per-request _meta envelope, so the server sees it on
every request; on a legacy connection it rides the initialize handshake.
Server code doesn't care which: require_client_extension(ctx, ...) and
ctx.session.check_client_capability(...) read the right source on both paths.client.session.send_request(...); Client
only grows first-class methods for spec verbs. send_request accepts any
Request subclass, so the vendor request passes as-is.tools/callThe one interceptive hook. Override intercept_tool_call to observe, short-circuit,
or veto a tool call:
--8<-- "docs_src/extensions/tutorial005.py"
params is the validated CallToolRequestParams: you get params.name and
params.arguments without touching raw JSON.call_next(ctx) runs the rest of the chain. Return its result unchanged (observe),
return something else (replace), or raise an MCPError (refuse).extensions=[...] is outermost.The hook wraps tools/call and nothing else. For every-message concerns, use
Middleware. That is what it is for.
A client extension is the same contract from the consuming side: a bundle of
client-side behaviour behind one identifier. Pass instances to
Client(extensions=[...]) and call tools normally:
--8<-- "docs_src/extensions/tutorial006.py"
call_tool("buy", ...) returns a plain CallToolResult, like every other call. What
the extension changed: the server may now answer buy with a receipt result
shape instead of a final result, and Receipts finishes it (here by redeeming the
receipt with a follow-up call) before call_tool returns. Nothing about the call
site moves.
Drop the extension and none of this exists: the server's gate refuses a client
that did not declare it (error -32021), and a claimed shape from a server that
skips the gate fails validation, exactly as the spec requires for an
unrecognized resultType. Off by default, on both ends of the wire.
To advertise an identifier with no client-side behaviour (the server gates on
the capability, the client does nothing, as in the search client above), use
advertise():
from mcp.client import advertise
client = Client(mcp, extensions=[advertise("com.example/search")])
Subclass ClientExtension and override only what you need. Three contribution
kinds, each with a default: settings(), claims(), and notifications().
--8<-- "docs_src/extensions/tutorial006.py"
claims() returns ResultClaims: a wire tag, the model that parses it, and the
resolver that finishes it. The model must pin the tag with
result_type: Literal["receipt"] and must not subclass the verb's core result
types; both are enforced when the claim is constructed. Vendor fields like
receipt_token ride the wire as-is: a substituted shape reaches the client
verbatim.ClaimContext; ctx.session is the
same public handle as client.session, so follow-ups are ordinary session calls.
It returns the verb's normal CallToolResult.settings() is the value advertised at ClientCapabilities.extensions[identifier],
read once at Client construction.notifications() declares vendor server notifications to observe:
def notifications(self) -> Sequence[NotificationBinding[Any]]:
return [NotificationBinding(method="notifications/receipts", params_type=ReceiptEvent, handler=self.on_receipt)]
The handler receives validated params one at a time, in dispatch order. It observes; it cannot veto or reply.
Two quiet rules. Claims are active on 2026-07-28 connections only, and the capability
ad follows them: on a legacy connection the claims dissolve and the identifier drops
out of the ad with them, so the client never advertises an extension whose shapes it
would reject. And when you want the claimed shape yourself instead of the resolver,
call client.session.call_tool(..., allow_claimed=True); without that flag, a
claimed shape reaching a session-tier caller raises UnexpectedClaimedResult.
An extension's own request methods need no client-side registration. A vendor request
type subclasses mcp_types.Request and goes through client.session.send_request,
as in Serving your own methods. One addition: when a
params key must ride the Mcp-Name header (extension specs such as tasks require
this for their verbs), the request type declares name_param:
--8<-- "docs_src/extensions/tutorial007.py"
The session mirrors params["jobId"] into Mcp-Name on every send path, and a
missing value fails loudly rather than silently omitting a required header.
The contribution surface is closed on purpose. On the server: settings, tools,
resources, methods, one tools/call interceptor. On the client: settings, result
claims, notification bindings. An extension cannot:
initialize is reserved by the runner outright); a notification
binding shadowed by core vocabulary goes quiet with a warning instead.MCPServer(...) or Client(...) returns, the extension
set is what it is.If you are fighting these walls, you are not writing an extension. You are writing
a fork. The walls are the feature: a user reading extensions=[Apps(), Stamps()]
knows everything those two can have touched.