All posts

Claude Connectors Tutorial: Build and Deploy a Connector to Claude

Abe Wheeler
Claude Connectors Claude Apps MCP Apps MCP App Framework ChatGPT Apps ChatGPT App Framework
Building an interactive Claude Connector.

Building an interactive Claude Connector.

TL;DR: A Claude Connector is an MCP server. Build one by creating tools that return data from your service, optionally add MCP App resources for UI, test locally, then connect to Claude via Settings > Connectors. This tutorial walks through the full flow.

Claude Connectors are MCP servers that extend what Claude can do. Over 200 connectors are listed in the Connectors Directory at claude.ai, from Google Drive and Slack to Figma and Canva. Some return data for Claude to reason over. Others render full interactive UIs inside the chat.

In this tutorial, you’ll build an interactive connector that renders a support ticket card inside Claude, test it locally, and deploy it to a real Claude session.

Two Types of Connectors

Before building, it helps to understand the two types. (For a deeper comparison, see Claude Connectors vs Claude Apps.)

Standard connectors expose tools that return data. Claude calls the tool, gets structured data back, and uses it in a text response. A standard connector linking Claude to your issue tracker would let Claude look up tickets and describe them in prose.

Interactive connectors also include MCP App resources: React components that render UI inside the chat. Instead of describing a ticket in text, Claude renders your component as a visual card with status badges, assignee avatars, and action buttons. In the Connectors Directory, interactive connectors are marked with an “Interactive” badge. Figma, Canva, Asana, and Slack are all interactive connectors.

Both types are MCP servers. An interactive connector is a standard connector with a UI layer on top. This tutorial builds the interactive kind.

Anatomy of an Interactive Connector

An interactive Claude Connector has three parts:

  1. A tool that Claude calls. The tool has a schema (what arguments it accepts) and a handler (what it does when called). The handler fetches data from your service and returns structured content.
  2. A resource (UI component) that renders the tool’s output inside the conversation. This is a React component that receives the tool’s structured content and displays it as a card, chart, form, or whatever UI makes sense.
  3. A simulation (optional, for testing). A JSON fixture that defines a reproducible tool state, so you can develop and test your UI without calling the real backend every time.

The tool links to the resource by name. When Claude calls the tool and the handler returns structuredContent, Claude renders the linked resource component with that data.

Prerequisites

You need Node.js 20 or later and pnpm. You do not need a Claude account for local development.

Step 1: Scaffold the Project

This tutorial uses sunpeak to scaffold the project because it sets up the full MCP server structure (tools, resources, simulations, inspector) in one command:

npx sunpeak new

Name your project and pick any starter resources. We’re building a new resource from scratch, so the selection doesn’t matter. cd into your project directory.

Step 2: Build the Resource (UI)

Create src/resources/ticket/ticket.tsx:

import { useToolData, SafeArea } from 'sunpeak';
import type { ResourceConfig } from 'sunpeak';

export const resource: ResourceConfig = {
  title: 'Ticket',
  description: 'Display a support ticket',
};

interface TicketData {
  id: string;
  title: string;
  status: 'open' | 'in_progress' | 'resolved';
  priority: 'low' | 'medium' | 'high';
  assignee: string;
  created: string;
  description: string;
}

const statusColors = {
  open: 'bg-yellow-100 text-yellow-800',
  in_progress: 'bg-blue-100 text-blue-800',
  resolved: 'bg-green-100 text-green-800',
};

const priorityColors = {
  low: 'bg-gray-100 text-gray-700',
  medium: 'bg-orange-100 text-orange-700',
  high: 'bg-red-100 text-red-700',
};

export function TicketResource() {
  const { output } = useToolData<unknown, TicketData>(undefined, undefined);

  if (!output) return null;

  return (
    <SafeArea className="p-5 font-sans max-w-md mx-auto">
      <div className="flex items-start justify-between mb-3">
        <div>
          <span className="text-xs text-gray-400 font-mono">{output.id}</span>
          <h1 className="text-lg font-bold mt-0.5">{output.title}</h1>
        </div>
        <span className={`px-2 py-0.5 rounded-full text-xs font-medium ${priorityColors[output.priority]}`}>
          {output.priority}
        </span>
      </div>

      <p className="text-sm text-gray-600 mb-4">{output.description}</p>

      <div className="flex items-center gap-3 text-sm">
        <span className={`px-2 py-0.5 rounded-full text-xs font-medium ${statusColors[output.status]}`}>
          {output.status.replace('_', ' ')}
        </span>
        <span className="text-gray-400">|</span>
        <span className="text-gray-600">{output.assignee}</span>
        <span className="text-gray-400">|</span>
        <span className="text-gray-400">{output.created}</span>
      </div>
    </SafeArea>
  );
}

The component receives ticket data via useToolData and renders it as a card with status and priority badges. SafeArea handles padding so the content doesn’t overlap with host UI chrome. See the resource docs for all config options.

Step 3: Build the Tool (Backend)

Create src/tools/show-ticket.ts:

import { z } from 'zod';
import type { AppToolConfig, ToolHandlerExtra } from 'sunpeak/mcp';

export const tool: AppToolConfig = {
  resource: 'ticket',
  title: 'Show Ticket',
  description: 'Look up a support ticket and display it',
  annotations: { readOnlyHint: true },
};

export const schema = {
  ticketId: z.string().describe('Ticket ID to look up (e.g. TICK-1234)'),
};

type Args = z.infer<z.ZodObject<typeof schema>>;

export default async function (args: Args, _extra: ToolHandlerExtra) {
  // In production, fetch from your ticket system API using args.ticketId
  return {
    structuredContent: {
      id: 'TICK-1234',
      title: 'Search results not loading on mobile',
      status: 'in_progress',
      priority: 'high',
      assignee: 'Sarah Chen',
      created: '2026-03-04',
      description:
        'Users on iOS Safari report that search results fail to render after the latest deploy. Affects approximately 12% of mobile traffic.',
    },
  };
}

The resource: 'ticket' field links this tool to the ticket resource. When Claude calls this tool, the structured content gets passed to your React component. The annotations field is required for Connectors Directory submission. Every tool must declare readOnlyHint: true (for read operations) or destructiveHint: true (for write/delete operations). Missing annotations cause about 30% of rejections.

See the tool docs for all config options.

Step 4: Add a Simulation (Test Data)

Create tests/simulations/show-ticket.json:

{
  "tool": "show-ticket",
  "userMessage": "Show me ticket TICK-1234",
  "toolInput": {
    "ticketId": "TICK-1234"
  },
  "toolResult": {
    "structuredContent": {
      "id": "TICK-1234",
      "title": "Search results not loading on mobile",
      "status": "in_progress",
      "priority": "high",
      "assignee": "Sarah Chen",
      "created": "2026-03-04",
      "description": "Users on iOS Safari report that search results fail to render after the latest deploy. Affects approximately 12% of mobile traffic."
    }
  }
}

Simulations are JSON fixtures that define a reproducible tool state: the tool input Claude would send and the output your handler would return. The inspector loads them automatically so you can develop your UI against known data without calling a real backend. You can also load simulations in Playwright tests via URL for automated E2E testing. For a complete walkthrough of simulations, see the MCP App tutorial.

Step 5: Test Locally

pnpm dev

Open http://localhost:3000. The sunpeak inspector opens with your connector running. Select Claude from the Host dropdown in the sidebar. Your ticket card renders inside Claude’s conversation chrome with the warm beige palette.

Switch to ChatGPT in the dropdown to verify it works there too. The same component renders in both hosts because both implement the MCP App standard. This is one of the main advantages of building on the MCP standard: your connector works across hosts without code changes.

No Claude account needed for any of this. The local inspector replicates both runtimes on localhost.

Step 6: Connect to a Real Claude Session

When you’re ready to test in a real Claude session, you need a publicly accessible URL. Claude cannot reach localhost.

Create a tunnel

Use ngrok to expose your local server:

ngrok http 8000

Copy the forwarding URL (something like https://abc123.ngrok-free.app).

Add the custom connector in Claude

  1. Open Claude Connectors.
  2. Click the + button in the input area and select Add a connector.
  3. Enter your ngrok URL with the /mcp path: https://abc123.ngrok-free.app/mcp
  4. Click Add.

Only Pro, Max, Team, and Enterprise plans can add custom connectors.

Use it in a conversation

Ask Claude: “Show me ticket TICK-1234.”

Claude calls your show-ticket tool, and your ticket card renders inside the chat.

How Claude handles resource bundles

Claude’s iframe sandbox blocks HTTP script sources, which means it can’t load resources from a Vite dev server. When Claude fetches your resource, it needs self-contained HTML with all JavaScript inlined.

sunpeak handles this automatically: it detects Claude’s user-agent and serves the pre-built production bundle. When you save a file change during development, sunpeak auto-rebuilds and sends a notifications/resources/list_changed notification so Claude re-fetches the updated resource. If you’re building without sunpeak, your server needs to handle this same pattern.

Step 7: Submit to the Connectors Directory

To distribute your connector to all Claude users, submit it to the Connectors Directory.

Requirements

Before submitting, check these requirements:

  • Transport: Streamable HTTP. Your server must be internet-accessible. SSE support may be deprecated.
  • Annotations: Every tool must include readOnlyHint or destructiveHint in annotations. Missing annotations cause about 30% of rejections.
  • Token limit: Tool results cannot exceed 25,000 tokens.
  • Timeout: Tool handlers must complete within 5 minutes (300 seconds).
  • Auth: If your connector requires authentication, use OAuth with user consent flow. Provide a test account for Anthropic’s reviewers. Pure client credentials flow (machine-to-machine without user interaction) is not supported. See the OAuth guide for details.
  • OAuth callback: Allowlist both https://claude.ai/api/mcp/auth_callback and https://claude.com/api/mcp/auth_callback as redirect URIs.

Annotations example

// Read-only tool
export const tool: AppToolConfig = {
  resource: 'ticket',
  title: 'Show Ticket',
  description: 'Look up a support ticket and display it',
  annotations: { readOnlyHint: true },
};
// Destructive tool
export const tool: AppToolConfig = {
  resource: 'ticket',
  title: 'Delete Ticket',
  description: 'Permanently delete a support ticket',
  annotations: { destructiveHint: true },
};

Submit

Fill out the Connectors Directory server review form. Anthropic reviews submissions manually. The Directory submission guide has more details on what reviewers look for.

Your Connector Works Everywhere

The connector you just built is an MCP server. It works with any MCP-compatible host, not just Claude. The same resource component and tool handler run in ChatGPT, Goose, VS Code, and any future host that implements the MCP App standard.

To verify cross-host rendering, you can write Playwright tests that load your simulations in the inspector with different host settings:

import { test, expect } from 'sunpeak/test';

test('renders ticket card', async ({ inspector }) => {
  const result = await inspector.renderTool('show-ticket', {});
  const app = result.app();
  await expect(app.locator('text=TICK-1234')).toBeVisible();
});

This kind of automated cross-host testing catches rendering differences before you deploy. See the testing guide for a full walkthrough.

Get Started

Documentation →
npx sunpeak new

Further Reading

Frequently Asked Questions

How do I build a Claude Connector?

A Claude Connector is an MCP server that exposes tools to Claude. Build an MCP server with tools that return data from your service, deploy it to a public URL, and add it via Claude Settings > Connectors > Add custom connector. For interactive connectors that render UI inside the chat, your server also needs to register MCP App resources.

What is the difference between a standard Claude Connector and an interactive one?

A standard connector returns data that Claude weaves into text responses. An interactive connector also includes MCP App resources that render UI (cards, dashboards, forms) inside the chat. Interactive connectors are marked with an "Interactive" badge in the Connectors Directory. Both are MCP servers.

How do I connect my MCP server to Claude?

Run your MCP server on a publicly accessible URL (use ngrok for development). In Claude, go to Settings > Connectors > Add custom connector, enter your server URL with /mcp path, and save. Enable the connector per conversation via the + button. Free users can add one custom connector. Pro, Max, Team, and Enterprise plans support more.

Do I need a paid Claude account to develop a Claude Connector?

No. You can build and test your connector locally without any Claude account. Use a local MCP inspector to verify tool behavior and UI rendering. You only need a Claude account when you want to connect your finished connector to a real Claude session.

How do I submit my connector to the Claude Connectors Directory?

Fill out the Connectors Directory server review form on the Anthropic website. All tools must include readOnlyHint or destructiveHint annotations, as missing annotations cause 30% of rejections. Your server must use Streamable HTTP transport and provide a test account if authentication is required.

Does my Claude Connector also work in ChatGPT?

Yes, if your connector is built on the MCP standard. MCP App resources render in any MCP-compatible host, so the same tools and UI components work in both Claude and ChatGPT without code changes.

Why does Claude need a pre-built bundle instead of Vite HMR?

Claude's iframe sandbox blocks HTTP script sources, so it cannot load from a local Vite dev server. Your framework needs to detect Claude's user-agent and serve a pre-built production bundle instead. On file changes, the server sends a notifications/resources/list_changed notification so Claude re-fetches the resource.

What are the requirements for the Claude Connectors Directory?

Your server must use Streamable HTTP transport (SSE support may be deprecated). All tools need readOnlyHint or destructiveHint annotations. Tool results cannot exceed 25,000 tokens. Tool handlers must complete within 5 minutes. If your connector requires authentication, use OAuth and provide a test account for review. Allowlist both claude.ai and claude.com callback URLs. Pure client credentials flow (machine-to-machine OAuth without user interaction) is not supported.