Skip to content

Talk to a Cloudflare Worker from Claude Desktop!

Notifications You must be signed in to change notification settings

threepointone/workers-mcp-server

 
 

Repository files navigation

Workers MCP Server

Talk to your Cloudflare Workers from Claude Desktop!

This is a proof-of-concept of writing a Model Context Protocol (MCP) Server in a Cloudflare Worker. This gives you a way to extend Claude Desktop (among other MCP clients) by invoking functions in your Worker, which gives you access to any Cloudflare or third-party binding.

You write worker code that looks like this:

export class ExampleWorkerMCP extends WorkerEntrypoint<Env> {
  /**
   * Generates a random number. This is extra random because it had to travel all the way to
   * your nearest Cloudflare PoP to be calculated which... something something lava lamps?
   *
   * @return {string} A message containing a super duper random number
   * */
  async getRandomNumber() {
    return `Your random number is ${Math.random()}`
  }
}

And, using the provided MCP proxy, your Claude Desktop can see & invoke these messages:

image

Yes, I know that Math.random() works the same on a Worker as it does on your local machine, but don't tell Claude 🤫

Neat! How do I play?

  1. Download Claude Desktop https://claude.ai/download
  2. Clone this repo.
    There's a few pieces of novel code that need to hang together to make this work, so for now the way to play with it is to clone this repo first. Then, from within this folder:
  3. pnpm install
  4. Check wrangler.json
    The current demo uses both the Email Routing API and Durable Objects. If you don't have access to these, or they're not enabled, comment out the relevant sections in wrangler.json or your deploy will fail.
  5. pnpm deploy:worker
    This takes your src/index.ts file and generates generated/docs.json from it, then wraps it in templates/wrapper.ts and builds & deploys it using Wrangler.
    • You'll also need to run pnpm update:secret the very first time you do this.
  6. pnpm install:claude <server-alias> <worker-url> <entrypoint-name>
    For me, that's pnpm install:claude workers-mcp-server https://workers-mcp-server.glen.workers.dev ExampleWorkerMCP
  7. Restart Claude Desktop You have to do this pretty often, but you definitely have to do it after running the install step above.

To iterate on your server, do the following:

  1. Make your change to src/index.ts
  2. pnpm deploy:worker
  3. (usually) Restart Claude Desktop.
    You have to do this whenever you add/remove/change your methods or any of the documentation (Claude doesn't detect changes while it's running). But if you're just updating the code within a method then pnpm deploy:worker is enough.

How it works

Separately to your MCP code inside src/index.ts, there are three pieces required to make this work:

1. Docs generation: scripts/generate-docs.ts

The MCP Specification separates the tools/list and tools/call operations into separate steps, and most MCP servers have naturally followed suit and separated their schema definition from the implementation. However, combining them provides a much better DX for the author.

I'm using ts-blank-space and jsdoc-api to parse the TS and emit the schema, slightly tweaked. This gives you LLM-friendly documentation at build time:

/**
 * Send a text or HTML email to an arbitrary recipient.
 *
 * @param {string} recipient - The email address of the recipient.
 * @param {string} subject - The subject of the email.
 * @param {string} contentType - The content type of the email. Can be text/plain or text/html
 * @param {string} body - The body of the email. Must match the provided contentType parameter
 * @return {Promise<string>} A success message.
 * @throws {Error} If the email fails to send, or if that destination email address hasn't been verified.
 */
async sendEmail(recipient: string, subject: string, contentType: string, body: string) {
  // ...
}
{
  "ExampleWorkerMCP": {
    "exported_as": "ExampleWorkerMCP",
    "description": null,
    "methods": [
      {
        "name": "sendEmail",
        "description": "Send a text or HTML email to an arbitrary recipient.",
        "params": [
          {
            "description": "The email address of the recipient.",
            "name": "recipient",
            "type": "string"
          },
          {
            "description": "The subject of the email.",
            "name": "subject",
            "type": "string"
          },
          {
            "description": "The content type of the email. Can be text/plain or text/html",
            "name": "contentType",
            "type": "string"
          },
          {
            "description": "The body of the email. Must match the provided contentType parameter",
            "name": "body",
            "type": "string"
          }
        ],
        "returns": {
          "description": "A success message.",
          "type": "Promise.<string>"
        }
      }
    ]
  }
}

This list of methods is very similar to the required MCP format for tools/list, but also gives us a list of the WorkerEntrypoint exports names to look up our service bindings later.

To iterate on your docs, run pnpm generate:docs:watch and you'll see the output change as you tweak your JSDoc in your src/index.ts (you'll need watchexec installed).

2. Public HTTP handler: templates/wrapper.ts

Since our WorkerEntrypoint is not directly accessible, we need something that defines a default export with a fetch() handler. This is what templates/wrapper.ts does.

This exposes a single endpoint, /rpc, which takes a JSON payload of { entrypoint: string, method: string, args?: any[] }, then directly translates that to:

// BINDINGS injected from wrangler.json
const binding_config = BINDINGS[entrypoint]
return await env[binding_config.binding][method](...args)

This means that you need an entry in your services: object in wrangler.json for each entrypoint you want to connect to, e.g.:

"services": [
  {
    "binding": "MCP",
    "service": "workers-mcp-server",
    "entrypoint": "ExampleWorkerMCP"
  }
]

If you want to start using multiple entrypoints, you'll need to call pnpm install:claude <server-alias> <worker-url> <entrypoint-name> for each one. Then, Claude will see them as separate MCP servers, and presumably invoke them differently than if they were all lumped together in one.

3. Local MCP proxy: scripts/local-proxy.ts

This file uses the @modelcontextprotocol/sdk library to build up a normal, local MCP server. This responds to tools/list by producing the data from docs.json for the specified entrypoint.

On tools/call, a .fetch call is made to the remote worker on the /rpc route, providing a Bearer token with the contents of generated/.shared-secret. The responses are then piped back to Claude.

Calling pnpm install:claude <server-alias> <worker-url> <entrypoint-name> adds a sever definition that point to this file in your claude_desktop_config.json:

{
  "mcpServers": {
    "<server-alias>": {
      "command": "<absolute-path-to>/node",
      "args": [
        "<project-dir>/node_modules/tsx/dist/cli.mjs",
        "<project-dir>/scripts/local-proxy.ts",
        "<server-alias>",
        "<worker-url>",
        "<entrypoint-name>"
      ]
    }
  }
}

In this way you can install as many of these as you like, as long as they each have a distinct <server-alias>.

Limitations

There are lots. This pizza is straight out of the oven. You may well burn your mouth.

  1. docs.json is only generated from src/index.ts. It doesn't currently crawl imports like a bundler, because no bundler I could find preserved comments in-place in order for me to run the docs generator afterwards.
  2. Documentation generation only works for class exports, and can't handle default exports yet. It can, at least, parse class X {}; export { X as Y }, but in general most people do export class X {} anyway so this is fine for now.
  3. The local proxy <-> remote proxy communication doesn't follow any particular RPC spec, but it probably should.
  4. Error handling, non-text return values, streaming responses, etc have not really been thought through or aren't yet supported in Claude.
  5. No wrangler dev support yet, but wrangler dev --remote should be possible so you don't have to deploy so often
  6. Following on from the above, the spec includes a notifications/tools/list_changed notification that should trigger Claude to refresh its list of the tools available, meaning fewer restarts of Claude Desktop. But I haven't implemented that yet.
  7. No support for Resources. It might be cool to be able to define the list of verified email addresses as a resource, for example. That would be a resource, right?
  8. The docs parsing doesn't yet use TS types to either augment or replace the need for @param blocks in the JSDoc
  9. The doc generation might be completely superfluous if someone was using a validator like zod or a schema library like typebox. However, I wanted build-time docs generation (i.e. static extraction) and wanted to be as generic as possible, so JSDoc will do for now.
  10. This works on my machine, but has only been tested on my machine.

Future ideas

Obviously, having Claude Desktop talk directly to the Worker would be ideal. Also, wrangler dev --remote support would be great: you could iterate on your worker without redeploying, but still access your production bindings.

The docs generator needs to be extracted into a library so we can publish changes, as it needs to grow in scope to be really useful, and likely incorporate other sources of data (d.ts files, zod schemas, etc).

Feedback & Contributions

Give it a try! Then, raise an issue or send a PR and we can discuss what needs to change. This is all very new, so it could really go in a lot of different directions. We'd love to hear from you!

About

Talk to a Cloudflare Worker from Claude Desktop!

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • TypeScript 100.0%