cloudflare-dns-cloudflared-mcp

cloudflare-dns-cloudflared-mcp

Self-hosted MCP server for administering Cloudflare DNS and cloudflared tunnels, enabling exposure of SSH hosts and web services with Google OAuth access control.

Category
Visit Server

README

cloudflare-dns-cloudflared-mcp

Version

Architecture

Self-hosted MCP server for administering Cloudflare DNS and cloudflared tunnel services — expose SSH hosts, web UIs, and other services on your home network through Cloudflare Tunnels with Google OAuth access control.

Runs as a Docker container on your own infrastructure. Connects to Claude Code or any MCP-compatible client via bearer-token-authenticated HTTP.


Tools

DNS

Tool Description
dns_list_zones List all zones in the account
dns_list_records List DNS records for a zone
dns_create_record Create a DNS record (A, AAAA, CNAME, MX, TXT, …)
dns_update_record Update an existing DNS record
dns_delete_record Delete a DNS record

Tunnels

Tool Description
tunnel_list List all Cloudflare Tunnels
tunnel_get Get tunnel details
tunnel_get_token Get connector token for cloudflared
tunnel_list_connections List active tunnel connections

Tunnel Services (workflow)

Tool Description
service_list List all services exposed across all tunnels
service_expose_ssh Expose an SSH host through a tunnel with browser-based access
service_expose_web Expose a web UI through a tunnel with access control
service_remove Remove a service — tears down ingress, DNS, and Access app

What service_expose_ssh and service_expose_web do

Each workflow tool wires up the full stack in one call:

  1. Tunnel ingress rule — maps the public hostname to the private backend service
  2. DNS CNAME — points subdomain.yourdomain.com[tunnel-id].cfargotunnel.com
  3. Cloudflare Access application — gates who can reach the service
  4. Access policy — allows specific Google accounts, with optional one-time PIN (OTP) for non-Google emails

Prerequisites

Cloudflare Tunnel

You need a running cloudflared tunnel connected to your home network. Install cloudflared on your home server and connect it via the Cloudflare Zero Trust dashboard. The tunnel must show as Online before exposing services through it.

Google OAuth identity provider

For Google-authenticated access, configure Google as an identity provider once in the Zero Trust dashboard under Settings → Authentication. This is a one-time manual setup — the MCP server manages per-service access policies, not the identity provider itself.


Quick start

1. Clone

git clone git@github.com:andrewkriley/cloudflare-dns-cloudflared-mcp.git
cd cloudflare-dns-cloudflared-mcp

2. Configure

cp .env.example .env

Edit .env and fill in:

Variable Description
CF_API_TOKEN Cloudflare API token (see permissions below)
CF_ACCOUNT_ID Your Cloudflare account ID
MCP_BEARER_TOKEN Shared secret for MCP client auth — generate with openssl rand -hex 32

3. Run

docker compose up -d

Check it's healthy:

curl http://localhost:3000/health
# {"status":"ok"}

4. Connect Claude Code

Run once to register the server:

claude mcp add cloudflare-admin --transport http \
  --header "Authorization: Bearer YOUR_MCP_BEARER_TOKEN" \
  http://localhost:3000/mcp

Replace YOUR_MCP_BEARER_TOKEN with the value from your .env.

5. Example usage

Ask Claude:

"Expose my Proxmox server at 192.168.1.100:8006 as proxmox.yourdomain.com through my home tunnel. Allow access for user@gmail.com."

Claude will call dns_list_zones, tunnel_list, then service_expose_web to wire everything up.


Secrets & Security

Environment variables

Variable Sensitivity Purpose
CF_API_TOKEN Secret Cloudflare API token — DNS + Tunnel + Zero Trust permissions
CF_ACCOUNT_ID Low Cloudflare account identifier
MCP_BEARER_TOKEN Secret Shared secret for MCP endpoint auth
MCP_PORT Low HTTP port (default: 3000)

All are loaded from .env via docker compose. The .env file is gitignored and must never be committed.

Creating the Cloudflare API token

Go to dash.cloudflare.com/profile/api-tokens and create a custom token with:

Scope Permission
Zone > DNS Edit
Account > Cloudflare Tunnel Edit
Account > Zero Trust Edit
Zone > Zone Read

Set a 90-day expiry at creation time.

Token rotation (every 90 days)

  1. Create a new token at dash.cloudflare.com/profile/api-tokens
  2. Update CF_API_TOKEN in .env
  3. Restart the container: docker compose restart
  4. Delete the old token in the Cloudflare dashboard

Generating a bearer token

openssl rand -hex 32

Security controls

Control What it does
Bearer token auth All /mcp requests require Authorization: Bearer <token>
Non-root container Process runs as unprivileged mcp user
Read-only filesystem Container root filesystem is read-only (tmpfs for /tmp)
No new privileges no-new-privileges:true prevents privilege escalation
/health is auth-free Returns {"status":"ok"} only — no sensitive data exposed

Development

Run locally without Docker

npm install
cp .env.example .env
# fill in .env
npm run dev

Build and test

npm run build          # compile TypeScript
npm run typecheck      # type-check without emitting
npm run lint           # ESLint
npm test               # unit tests (mocked, no credentials needed)
npm run test:integration  # integration tests (requires env vars — runs in CI)

Docker commands

docker compose up -d             # start in background
docker compose logs -f           # tail logs
docker compose restart           # restart after .env change
docker compose down              # stop and remove container
docker compose build --no-cache  # force rebuild image

CI

Every push and pull request runs:

Check Tool Purpose
Secret scanning Gitleaks Detects accidentally committed tokens
TypeScript check tsc --noEmit Strict compile-time correctness
ESLint eslint Code quality and style
Unit tests Vitest Mocked tests — workflow tool logic
Dependency audit npm audit --audit-level=high Flags high/critical vulnerabilities
Docker build docker/build-push-action Verifies image builds successfully
Tunnel integration tests Vitest + real cloudflared Full lifecycle against a real ephemeral tunnel
VERSION ↔ package.json npm run verify-version Keeps root VERSION and package.json in sync
Version bump (PRs) Compare VERSION to base Every PR must bump VERSION when present on base (see VERSIONING.md)

All checks above are required for a PR to merge to main once enabled in branch protection. Pushing a tag v*.*.* runs the Release workflow (GitHub Release + GHCR image + latest git tag).


Known limitations

Browser-rendered SSH and short-lived certificates

service_expose_ssh sets up the correct server-side configuration for Cloudflare short-lived SSH certificates (TrustedUserCAKeys, AuthorizedPrincipalsFile, SSH CA). However, Cloudflare's browser-rendered SSH terminal uses libssh2 1.9.0, which does not support OpenSSH certificate authentication. Certificate support was added in libssh2 1.11.0 (2023).

Effect: When accessing an SSH service via the browser terminal, the browser prompts for a private key instead of logging in automatically with a short-lived cert.

Short-lived certs work correctly for native SSH client access via the cloudflared ProxyCommand:

Host ssh.yourdomain.com
  ProxyCommand cloudflared access ssh --hostname %h

Workarounds for browser-only access (e.g. when outbound SSH is blocked on the client network):

Option A — Enable password auth on the backend sshd. The SSH port is not publicly exposed (only reachable via the tunnel, gated by Cloudflare Access). Password auth behind Google OAuth is an acceptable trade-off:

# On the SSH backend host
sudo sed -i 's/^PasswordAuthentication no/PasswordAuthentication yes/' /etc/ssh/sshd_config
sudo systemctl reload ssh

Option B — Deploy a web terminal (wetty). Run wetty on the backend host and expose it as a web service. The browser connects to wetty over HTTPS; wetty connects to sshd on localhost. No libssh2 limitation applies.

docker run -d --restart unless-stopped \
  --name wetty \
  -p 3000:3000 \
  wettyoss/wetty --ssh-host localhost --ssh-port 22 --base /

Then expose via service_expose_web with backend_port: 3000, backend_protocol: http. Cloudflare Access gates access as normal.


Versioning

The canonical version is the root VERSION file (same pattern as splunk-lab). Run npm run sync-version to copy it into package.json. Pushing a tag v*.*.* triggers a GitHub Release and publishes the Docker image to GHCR.

See VERSIONING.md for the full workflow, CI rules, and tag format.


Contributing

main is protected — all changes via PR from dev. CI must pass before merge. Direct pushes to main are blocked.

Recommended Servers

playwright-mcp

playwright-mcp

A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.

Official
Featured
TypeScript
Magic Component Platform (MCP)

Magic Component Platform (MCP)

An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.

Official
Featured
Local
TypeScript
Audiense Insights MCP Server

Audiense Insights MCP Server

Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.

Official
Featured
Local
TypeScript
VeyraX MCP

VeyraX MCP

Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.

Official
Featured
Local
graphlit-mcp-server

graphlit-mcp-server

The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.

Official
Featured
TypeScript
Kagi MCP Server

Kagi MCP Server

An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.

Official
Featured
Python
E2B

E2B

Using MCP to run code via e2b.

Official
Featured
Neon Database

Neon Database

MCP server for interacting with Neon Management API and databases

Official
Featured
Exa Search

Exa Search

A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.

Official
Featured
Qdrant Server

Qdrant Server

This repository is an example of how to create a MCP server for Qdrant, a vector search engine.

Official
Featured