MCP Tutorial
A minimal Model Context Protocol (MCP) tutorial built with theMCP Python SDK (mcp[cli]). It contains a smallFastMCP server and shows how to make Claude Code aware of it through project configuration — withoutrunning claude mcp add.
Contents
- The server
- Prerequisites
- Running the server standalone
- Registering the server with a client
- Claude Code
- Option A — stdio (Claude Code launches the server)
- Option B — HTTP transport
- Reload and verify
- GitHub Copilot (VS Code)
- stdio
- HTTP transport
- Start and verify
- Claude Code
- Approving servers and auto-approving tools (Claude Code)
- Step 1 — Approve the server
- Step 2 — Auto-approve tool calls
- Debugging with the MCP Inspector
- Configuration reference
The server
The example server lives in examples/snippets/servers/fastmcp_quickstart.pyand exposes one of each MCP primitive:
| Primitive | Name | Description |
|---|---|---|
| Tool | add(a, b) |
Adds two numbers |
| Resource | greeting://{name} |
Returns a personalized greeting |
| Prompt | greet_user(name, style) |
Generates a greeting prompt |
Prerequisites
- uv for running the project
- Dependencies are declared in
pyproject.toml(mcp[cli]>=1.28.0)
Install dependencies:
uv sync
Running the server standalone
To run the server directly over streamable HTTP (listens on http://localhost:8000/mcp):
uv run examples/snippets/servers/fastmcp_quickstart.py
This is only required for the HTTP transport option below. With thestdio option, Claude Code launches the server foryou, so you don't need to run this manually.
Registering the server with a client
Each MCP client discovers servers from its own config file. The server configuration is nearly identicalacross clients — the differences are which file holds it and the top-level key name:
| Client | Config file | Top-level key |
|---|---|---|
| Claude Code | .mcp.json |
mcpServers |
| GitHub Copilot | .vscode/mcp.json |
servers |
Both files are committed to the repo, so anyone who clones it gets the same server configuration.
Claude Code
Claude Code learns about project-scoped MCP servers from a .mcp.json file at the repository root.This file is the declarative equivalent of claude mcp add — it's committed to the repo, so anyone whoclones it gets the same server configuration.
Pick one of the two transport options below.
Option A — stdio (Claude Code launches the server)
Because the server lives in this project, you can let Claude Code spawn it on demand. There's no port andnothing to keep running separately.
.mcp.json:
{
"mcpServers": {
"demo-server": {
"command": "uv",
"args": ["run", "examples/snippets/servers/fastmcp_quickstart.py"]
}
}
}
Note: For stdio, the server script must use the stdio transport. FastMCP uses stdio by default whenyou call
mcp.run()with no arguments, so change the last line of the script tomcp.run()(or run astdio-specific entry point).
Option B — HTTP transport
If you'd rather run the server yourself as a long-running HTTP process, point .mcp.json at its URLinstead. First start the server (see Running the server standalone),then use:
.mcp.json:
{
"mcpServers": {
"demo-server": {
"type": "http",
"url": "http://localhost:8000/mcp"
}
}
}
With this option the server must already be running before Claude Code connects.
Note: For HTTP, the server script must use the streamable HTTP transport. In
fastmcp_quickstart.pythe defaultmcp.run()iscommented alongsidemcp.run(transport="streamable-http")— swap the active line so the script runsover HTTP:if __name__ == "__main__": mcp.run(transport="streamable-http") # mcp.run() # default transport is stdio
Reload and verify
Restart Claude Code (or run
/mcpin an interactive session).From the terminal, check status with:
claude mcp list claude mcp get demo-serverThe server should report ✔ Connected and expose the
addtool, thegreeting://{name}resource,and thegreet_userprompt.
GitHub Copilot (VS Code)
VS Code discovers workspace MCP servers from .vscode/mcp.json. Note the top-levelkey is servers (not mcpServers), and each entry declares its transport with type.
Pick one of the two transport options below.
stdio
VS Code launches the server for you — no port, nothing to keep running separately:
.vscode/mcp.json:
{
"servers": {
"demo-server": {
"type": "stdio",
"command": "uv",
"args": ["run", "examples/snippets/servers/fastmcp_quickstart.py"]
}
}
}
Note: As with Claude Code, the script must use the stdio transport (
mcp.run()).
HTTP transport
Run the server yourself first (see Running the server standalone), thenpoint VS Code at its URL:
.vscode/mcp.json:
{
"servers": {
"demo-server": {
"type": "http",
"url": "http://localhost:8000/mcp"
}
}
}
Note: For HTTP, the script must use
mcp.run(transport="streamable-http")(see the Claude CodeHTTP note).
Start and verify
- Open
.vscode/mcp.json— VS Code shows a Start action above each server entry.You can also run MCP: List Servers from the Command Palette to start/stop/inspect them. - Open the Copilot Chat view and switch to Agent mode (tools are only available in agent mode).
- Click the tools (🛠) icon to confirm
demo-serveris listed, then ask Copilot to use theaddtool.
Approving servers and auto-approving tools (Claude Code)
Claude Code has two independent safety gates for project MCP servers, both configured in.claude/settings.json (shared, committed) or .claude/settings.local.json (personal, git-ignored):
- Server approval — whether a server from
.mcp.jsonis allowed to load at all. - Tool-call approval — whether Claude may run a server's tools without prompting you each time.
This section walks through both, using the lifespan-demo server (which exposes the query_db tool) asthe example.
Step 1 — Approve the server
Servers declared in .mcp.json start as ⏸ Pending approval as a safety measure — a cloned repo can'tauto-run servers without your consent. Grant approval declaratively with enabledMcpjsonServers insteadof clicking through the interactive prompt:
{
"enabledMcpjsonServers": ["demo-server", "lifespan-demo"]
}
Or trust every server defined in .mcp.json at once:
{
"enableAllProjectMcpServers": true
}
After this, the server loads and its tools become visible — but Claude will still ask permission eachtime it wants to call one.
Step 2 — Auto-approve tool calls
To stop the per-call prompts, add a permissions.allow rule. The MCP rule format is:
| Rule | Matches |
|---|---|
mcp__lifespan-demo |
Every tool from the lifespan-demo server |
mcp__lifespan-demo__query_db |
Only the query_db tool |
Note: the tool part does not support
*wildcards — use the server-only form (mcp__<server>) tocover all tools, or name a specific tool (mcp__<server>__<tool>).
Combined with Step 1, .claude/settings.json looks like:
{
"enabledMcpjsonServers": ["demo-server", "lifespan-demo"],
"permissions": {
"allow": ["mcp__lifespan-demo"]
}
}
Reload Claude Code (restart or /mcp), then ask:
Use the query_db tool to run
SELECT * FROM users
It runs without a permission prompt and returns Result of 'SELECT * FROM users'.
Scope tip: keep server approval (
enabledMcpjsonServers) in the sharedsettings.jsonsoteammates get the server, but consider putting auto-approval (permissions.allow) insettings.local.jsonif you don't want to silently grant tool execution to everyone who clones the repo.
Debugging with the MCP Inspector
The MCP Inspector is an interactive, browser-based tool for poking at a server directly — withoutwiring it into Claude Code or Copilot. It's the quickest way to call tools by hand and watch logs andprogress notifications live.
Launch it against any server in this repo:
uv run mcp dev examples/snippets/servers/<server_file>.py
uv run mcp dev examples/snippets/servers/basic_tool.py
What this does:
uv runruns inside the project virtualenv (somcpand your deps are available).mcp devstarts the server over stdio and attaches the Inspector to it.- It prints a
localhostURL — open it in a browser.
The Inspector uses two ports: 6274 (the web UI) and 6277 (the proxy). In the UI you can:
- List and call tools — e.g. run
long_running_taskwith atask_nameandsteps. - Browse resources and prompts.
- Watch logs and progress notifications live — the best way to see
Contextcalls(ctx.info,ctx.debug,ctx.report_progress) as they happen.
Requires Node.js. The Inspector is a Node package (
@modelcontextprotocol/inspector) launched vianpx; the first run downloads it.Auth token. Recent versions print a URL that includes a session token, e.g.
http://localhost:6274/?MCP_PROXY_AUTH_TOKEN=…— use that full link from the terminal.Port already in use?
❌ Proxy Server PORT IS IN USE at port 6277means an Inspector is alreadyrunning — just open http://localhost:6274, or stop the existing one first. It's a long-runningprocess; stop it withCtrl-C.
Configuration reference
| File | Purpose | Committed? |
|---|---|---|
.mcp.json |
Declares the project MCP server for Claude Code | Yes (shared) |
.vscode/mcp.json |
Declares the workspace MCP server for Copilot | Yes (shared) |
.claude/settings.json |
Approves servers + auto-approves tools for the team | Yes (shared) |
.claude/settings.local.json |
Approves servers + auto-approves tools just for you | No (git-ignored) |