This guide shows the canonical end-to-end wiring for protecting an HTTP MCP
server with attesto_mcp: mount the metadata discovery routes, protect the MCP
endpoint with one plug, and require the scopes the endpoint needs. Every step is
copy-pasteable.
The pieces fit together so that the RFC 9728 resource identifier a client
discovers and the resource_metadata challenge the server returns on a 401
always agree, because both derive from the same request origin and resource
path.
1. Attesto config
attesto_mcp delegates token, DPoP, and mTLS verification to Attesto, so the
host supplies an Attesto.Config (or a zero-arity function returning one).
defmodule MyApp.Attesto do
def config do
Attesto.Config.new(
issuer: "https://auth.example.com",
audience: "https://mcp.example.com/mcp",
keystore: MyApp.Attesto.Keystore
)
end
end2. DPoP replay protection
DPoP proof replay protection is required for protected-resource requests. Wire a
shared :replay_check callback (an ETS store for a single node, a
database-backed store for a cluster). Without it, DPoP requests fail closed
through Attesto.
replay_check = &MyApp.DPoPReplay.check_and_record/23. Mount discovery routes
use AttestoMCP.Router adds attesto_mcp_protected_resource_metadata/2, which
serves /.well-known/oauth-protected-resource/<path> for each resource (plus a
backwards-compatible root /.well-known/oauth-protected-resource).
defmodule MyAppWeb.Router do
use Phoenix.Router
use AttestoMCP.Router
pipeline :api do
plug :accepts, ["json"]
end
scope "/" do
pipe_through :api
attesto_mcp_protected_resource_metadata "/mcp",
scopes: [AttestoMCP.Scopes.tools_call()]
end
# ... protected endpoint below
endServing more than one MCP server is one call per resource. Each gets its own metadata document; the root route resolves to the first declared resource.
attesto_mcp_protected_resource_metadata "/mcp/foo", scopes: ["foo:mcp:tools:call"]
attesto_mcp_protected_resource_metadata "/mcp/bar", scopes: ["bar:mcp:tools:call"]4. Protect the endpoint with one plug
AttestoMCP.Plug.ProtectResource composes authentication and scope enforcement
into one correctly ordered, halt-respecting plug. The :resource it is given is
the same path mounted for discovery above, so the resource_metadata challenge
on a 401 points at the route from step 3.
pipeline :mcp do
plug :accepts, ["json", "sse"]
plug AttestoMCP.Plug.ProtectResource,
config: &MyApp.Attesto.config/0,
replay_check: &MyApp.DPoPReplay.check_and_record/2,
resource: "/mcp",
scopes: [AttestoMCP.Scopes.tools_call()],
principal: fn claims, sender ->
MyApp.Principals.from_token(claims, sender)
end
end
scope "/" do
pipe_through :mcp
forward "/mcp", MyApp.MCPServerPlug
endAfter authentication, downstream code can read:
conn.assigns.attesto_mcp_claimsconn.assigns.attesto_mcp_scopesconn.assigns.attesto_mcp_senderconn.assigns.attesto_mcp_principal, if:principalis configured
5. mTLS-bound tokens (optional)
For mTLS sender-constrained tokens, supply certificate context from the TLS
layer. The callback returns the DER-encoded certificate the TLS layer already
authenticated, or nil when none was presented.
plug AttestoMCP.Plug.ProtectResource,
config: &MyApp.Attesto.config/0,
resource: "/mcp",
scopes: [AttestoMCP.Scopes.tools_call()],
cert_der: fn conn -> MyApp.TLS.client_certificate_der(conn) end6. Test the binding contract
AttestoMCP.Test.DPoPAssertions ships ExUnit assertions that drive your wired
pipeline and prove DPoP binding holds: a DPoP-bound token presented as a plain
Bearer is rejected, and the same token presented with a valid proof is accepted.
defmodule MyAppWeb.MCPAuthTest do
use ExUnit.Case
import AttestoMCP.Test.DPoPAssertions
setup do
%{config: AttestoMCP.Test.Factory.config()}
end
test "the MCP pipeline enforces DPoP binding", %{config: config} do
plug = fn conn ->
opts =
AttestoMCP.Plug.ProtectResource.init(
config: config,
replay_check: AttestoMCP.Test.DPoPReplay.callback(),
scopes: [AttestoMCP.Scopes.tools_call()]
)
AttestoMCP.Plug.ProtectResource.call(conn, opts)
end
assert_dpop_bound_bearer_rejected(plug, config, scopes: [AttestoMCP.Scopes.tools_call()])
assert_dpop_proof_accepted(plug, config, scopes: [AttestoMCP.Scopes.tools_call()])
end
end