Deploy MCP tunnels with Helm
Install the MCP tunnel stack on a Kubernetes cluster using the Anthropic Helm chart.
MCP tunnels is a research preview feature. Request access to try it.
The Anthropic Helm chart installs the MCP tunnel stack as a single Deployment and attaches it to the tunnel you created in the Console.
Before you begin
You need:
- A tunnel created in the Console. Follow Create a tunnel and record the tunnel ID (
tnl_...). - A way for the chart to authenticate to the Tunnels API.
- Programmatic access (recommended). The chart's setup Job authenticates through Workload Identity Federation, fetches the tunnel token, generates a CA, registers it with Anthropic, and stores everything in a Secret. You'll need a federation rule scoped to
org:manage_tunnels. - Manual. Skip programmatic access. You'll get the tunnel token from the Console, generate a CA and server certificate yourself, register the CA in the Console, and supply the credentials to the cluster as Secrets.
- Programmatic access (recommended). The chart's setup Job authenticates through Workload Identity Federation, fetches the tunnel token, generates a CA, registers it with Anthropic, and stores everything in a Secret. You'll need a federation rule scoped to
- A Kubernetes cluster you can deploy to with
helmandkubectl. The Without programmatic access tab also usesopenssl(1.1.1 or later). - Outbound network connectivity from the cluster to
api.anthropic.com(443 TCP) and the tunnel edge (7844 TCP and UDP). See the full network requirements. - One or more MCP servers running and reachable from the cluster on the addresses you'll configure under
gateway.config.routes. If you don't have one yet, use the sample server below.
Optional: Use a sample MCP server
If you don't have an MCP server available for testing, use this minimal one:
kubectl create namespace mcp-tunnel --dry-run=client -o yaml | kubectl apply -f -
kubectl -n mcp-tunnel apply -f - <<'EOF'
apiVersion: v1
kind: ConfigMap
metadata:
name: hello-mcp-src
data:
hello_server.py: |
from mcp.server.fastmcp import FastMCP
mcp = FastMCP("hello-server", host="0.0.0.0", port=9000)
@mcp.tool()
def hello(name: str = "world") -> str:
"""Say hello to someone."""
return f"Hello, {name}!"
if __name__ == "__main__":
mcp.run(transport="streamable-http")
---
apiVersion: apps/v1
kind: Deployment
metadata:
name: hello-mcp
spec:
replicas: 1
selector:
matchLabels: { app: hello-mcp }
template:
metadata:
labels: { app: hello-mcp }
spec:
containers:
- name: hello-mcp
image: python:3.13-slim
command: ["sh", "-c", "pip install --quiet mcp && python /app/hello_server.py"]
volumeMounts:
- { name: src, mountPath: /app }
ports:
- { containerPort: 9000 }
volumes:
- name: src
configMap: { name: hello-mcp-src }
---
apiVersion: v1
kind: Service
metadata:
name: hello-mcp
spec:
selector: { app: hello-mcp }
ports:
- { port: 9000, targetPort: 9000 }
EOF
The Install steps below note where to add the corresponding route.
Install
The chart's setup Job exchanges the cluster's projected ServiceAccount token through your federation rule, fetches the tunnel token, generates a CA and server certificate, and registers the CA with Anthropic. A daily CronJob renews the server certificate as needed. No manual secret handling.
Set up Workload Identity Federation for the cluster
Follow Use WIF with Kubernetes to register your cluster's OIDC issuer and create a federation rule. The chart's setup Job runs as ServiceAccount
<release>-setupin the release namespace, so use these values when creating the rule (assuming release namemcp-tunnelin namespacemcp-tunnel, which the rest of this guide uses; substitute yours):Field Value Subject system:serviceaccount:mcp-tunnel:mcp-tunnel-setupAudience api.anthropic.com(no scheme). This is the chart's default and must match the rule's audience byte-for-byte. If your rule hashttps://api.anthropic.com(the Console's suggestion), setapi.wif.audienceto that value.Scope org:manage_tunnelsIf the tunnel is in a workspace other than the organization's default, also add the rule's service account as a member of that workspace under Settings > Workspaces (the Tunnels API authorizes against the service account's workspace memberships).
Note the rule's ID (
fdrl_...); you'll set it asapi.wif.federationRuleId.NoteThe daily certificate-renewal CronJob uses a separate ServiceAccount (
<release>-cert-renew) but does not call the Tunnels API; it renews the certificate locally and only needs Kubernetes RBAC, which the chart grants. The federation rule does not need to cover it.Fetch the default values
helm show values \ oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \ --version 1.0.0 > values.yamlConfigure tunnel attachment and routes
Edit
values.yamland set theapi.wif.*keys with the tunnel ID, federation rule ID, and organization ID, plus aroutesentry for each upstream MCP server:api: wif: tunnelId: "tnl_..." federationRuleId: "fdrl_..." organizationId: "00000000-0000-0000-0000-000000000000" # Set when the tunnel is in a non-default workspace and the # rule's service account is a member of that workspace. # workspaceId: "wrkspc_..." gateway: config: routes: docs: http://docs-mcp.internal:8080 search: http://search-mcp.internal:8080With these routes, Claude reaches the servers at
docs.<your-tunnel-domain>andsearch.<your-tunnel-domain>. Some managed Kubernetes distributions allocate the Service CIDR outside the standard private ranges; if your routes target in-cluster Services, addgateway.config.upstream.allowed_ipshere per Upstream IP validation.NoteIf you're using the sample MCP server, set
routestoecho: http://hello-mcp:9000instead.Review the rendered manifests
Render the chart and review the output according to your organization's vetting practices:
helm template mcp-tunnel \ oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \ --version 1.0.0 \ -n mcp-tunnel \ -f values.yaml > rendered.yamlInstall
helm install mcp-tunnel \ oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \ --version 1.0.0 \ --namespace mcp-tunnel --create-namespace \ -f values.yamlThe setup Job runs as a Helm pre-install hook, so
helm installblocks until it completes; on success Helm deletes the Job automatically. Ifhelm installfails with a hook error, see Setup Job authentication failures.WarningThe
api.wif.*values are identifiers, not secrets, so storing them in Helm release-history Secrets is not a risk. The sensitive data at rest is themcp-tunnelSecret the setup Job creates, which holds the tunnel token and TLS private keys. Apply your organization's standard practices for protecting Kubernetes Secrets to this namespace.
In this mode (setup.enabled: false) the chart makes no API calls; there is no setup Job or cert-renew CronJob. Use this path if you'd rather not set up Workload Identity Federation.
Get the tunnel token and domain
Create the tunnel and get the tunnel token from the Console.
NoteRecord the tunnel domain from the detail page. You'll set it as
gateway.config.tunnel_domain.Generate a CA and server certificate
The proxy listens on plain WebSocket; the inner TLS handshake happens inside that stream using the certificate you generate here. The server certificate's SAN must include
*.<tunnel-domain>per the certificate requirements.export TUNNEL_DOMAIN=YOUR_TUNNEL_DOMAIN_HERE mkdir -p mcp-tunnel/data cd mcp-tunnel # Self-signed CA. Explicit extensions so it satisfies the certificate # requirements regardless of distro openssl.cnf defaults. openssl req -x509 -newkey rsa:2048 -nodes \ -keyout data/ca.key -out data/ca.crt \ -days 3650 -subj "/CN=mcp-tunnel-ca" \ -addext "basicConstraints=critical,CA:TRUE" \ -addext "keyUsage=critical,keyCertSign,cRLSign" \ -addext "subjectKeyIdentifier=hash" # Extension file for the server certificate. Using -extfile (instead of # -copy_extensions, which is OpenSSL 3.0+ only) keeps this working on # OpenSSL 1.1.x. cat > data/tls.ext <<EOF subjectAltName = DNS:${TUNNEL_DOMAIN},DNS:*.${TUNNEL_DOMAIN} authorityKeyIdentifier = keyid,issuer extendedKeyUsage = serverAuth EOF # Server certificate signed by the CA openssl req -newkey rsa:2048 -nodes \ -keyout data/tls.key -out /tmp/server.csr \ -subj "/CN=${TUNNEL_DOMAIN}" openssl x509 -req -in /tmp/server.csr \ -CA data/ca.crt -CAkey data/ca.key -CAcreateserial \ -out data/tls.crt -days 90 \ -extfile data/tls.extRegister
data/ca.crtin the Console. Keepdata/ca.keysomewhere durable and secure; you'll need it to sign a fresh server certificate at renewal time.Create the two Secrets
The chart reads specific keys; the Secret names are configurable but the keys are not. The first line is idempotent (the sample MCP server section already creates the namespace).
kubectl create namespace mcp-tunnel --dry-run=client -o yaml | kubectl apply -f - kubectl -n mcp-tunnel create secret generic mcp-tunnel-token \ --from-literal=tunnel-token='eyJ...' kubectl -n mcp-tunnel create secret generic mcp-tunnel-cert \ --from-file=tls.crt=data/tls.crt \ --from-file=tls.key=data/tls.keyFetch the default values
helm show values \ oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \ --version 1.0.0 > values.yamlSet values for external mode
Edit
values.yamland set the keys below:setup: enabled: false external: tunnelTokenSecretName: mcp-tunnel-token # must contain key: tunnel-token serverCertSecretName: mcp-tunnel-cert # must contain keys: tls.crt, tls.key gateway: config: # Required in external mode. In managed mode the chart injects this from # the Secret as a -tunnel-domain flag; in external mode you set it here. tunnel_domain: YOUR_TUNNEL_DOMAIN_HERE routes: docs: http://docs-mcp.internal:8080 search: http://search-mcp.internal:8080Some managed Kubernetes distributions allocate the Service CIDR outside the standard private ranges; if your routes target in-cluster Services, add
gateway.config.upstream.allowed_ipshere per Upstream IP validation.NoteIf you're using the sample MCP server, set
routestoecho: http://hello-mcp:9000instead.Review the rendered manifests
helm template mcp-tunnel \ oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \ --version 1.0.0 \ -n mcp-tunnel \ -f values.yaml > rendered.yamlInstall
helm install mcp-tunnel \ oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \ --version 1.0.0 \ --namespace mcp-tunnel \ -f values.yaml
Verify the deployment
Verify end to end from Anthropic's side: use https://<route>.<your-tunnel-domain>/ (where `` is whatever the upstream serves at) in a Managed Agent session or a Messages API request. See Use the tunneled MCP servers for the request shapes.
If that fails, check the pod logs (kubectl -n mcp-tunnel logs deploy/mcp-tunnel -c mcp-proxy and -c cloudflared) and consult Troubleshooting.
Optional configuration
Restrict egress with NetworkPolicy
Ingress to the proxy pod is denied by default (networkPolicy.ingress.enabled: true). To additionally restrict pod egress, set networkPolicy.egress.enabled: true and populate networkPolicy.egress.mcpServers with pod label selectors or CIDR ranges that cover your upstream MCP servers; cloudflared egress to the tunnel edge is allowed through networkPolicy.egress.cloudflaredEgressCIDRs.
Tune the proxy
Fields under gateway.config.* pass through to the proxy configuration file. Common adjustments include upstream.allowed_ips, log_level, and upstream.tls. See the proxy configuration reference for the full field list. The chart force-sets listen_addr, tls.cert_file, and tls.key_file; setting them in gateway.config has no effect.
Supply your own OIDC token
By default the chart projects a Kubernetes ServiceAccount token for the setup Job. To present a token from a different identity provider (SPIFFE, Vault, a cloud-SDK sidecar), mount it with setup.extraVolumes / setup.extraVolumeMounts and point api.wif.tokenFile at the mount path. The setup binary reads the token from ANTHROPIC_IDENTITY_TOKEN_FILE, which the chart sets to that path.
Upgrades
Always pass --version to helm upgrade so you don't pull a newer chart unexpectedly.
Change configuration
For routine changes such as routes, replica count, or NetworkPolicy:
helm upgrade mcp-tunnel \
oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \
--version 1.0.0 \
-n mcp-tunnel \
-f values.yaml
Maintain a complete values.yaml rather than relying on --reuse-values. Helm's deep-merge behavior can silently fail to remove deleted routes.
Rotate the tunnel token
With programmatic access, increment tunnel.tokenVersion in values.yaml and upgrade with --set setup.force=true. The setup hook only re-runs on upgrades when forced:
helm upgrade mcp-tunnel \
oci://us-docker.pkg.dev/anthropic-public-registry/charts/mcp-tunnel \
--version 1.0.0 \
-n mcp-tunnel \
-f values.yaml \
--set setup.force=true
The setup Job authenticates with Workload Identity Federation; there is no API token to revoke.
Without programmatic access, click Rotate token on the tunnel detail page in the Console, then update the mcp-tunnel-token Secret:
kubectl -n mcp-tunnel create secret generic mcp-tunnel-token \
--from-literal=tunnel-token='eyJ...' --dry-run=client -o yaml | kubectl apply -f -
kubectl -n mcp-tunnel rollout restart deploy/mcp-tunnel
Clicking Rotate token invalidates the current token immediately. Until the Secret is updated and the rollout completes, any pod that restarts with the old token (eviction, node drain, OOM) cannot reconnect. Update the Secret promptly after rotating; for stricter availability requirements, use programmatic access so the chart handles the rotation atomically.
Certificate renewal
The chart provides automation, but you remain responsible for monitoring expiry and confirming renewal completes.
With programmatic access, certificate renewal is automatic. The chart deploys a CronJob (<release>-cert-renew, daily at serverCert.cronSchedule, default 0 0 * * * UTC) that runs setup renew-cert and only renews when the certificate is within serverCert.renewBefore (default 30 days) of expiry. Renewal is local: it signs a fresh certificate with the CA stored in the Secret, makes no API calls, and only needs the Kubernetes RBAC the chart grants. The proxy hot-reloads the certificate from the Secret mount, so no Deployment restart is needed.
Without programmatic access there is no CronJob. From inside the mcp-tunnel/ directory you kept after install, sign a fresh server certificate with the existing CA (do not regenerate the CA):
export TUNNEL_DOMAIN=YOUR_TUNNEL_DOMAIN_HERE
openssl req -new -key data/tls.key -out /tmp/server.csr \
-subj "/CN=$\{TUNNEL_DOMAIN\}"
openssl x509 -req -in /tmp/server.csr \
-CA data/ca.crt -CAkey data/ca.key -CAcreateserial \
-out data/tls.crt -days 90 -extfile data/tls.ext
kubectl -n mcp-tunnel create secret generic mcp-tunnel-cert \
--from-file=tls.crt=data/tls.crt --from-file=tls.key=data/tls.key \
--dry-run=client -o yaml | kubectl apply -f -
The proxy hot-reloads the certificate from the Secret mount.