🤖 Agents IA

agent-mcp-server-builder

Création de serveurs MCP (Model Context Protocol) pour exposer des outils, ressources et prompts aux LLMs.

⚡ Installation & lancement en 1 commande

Copiez-collez dans votre terminal : le skill s'installe dans ~/.claude/skills et Claude Code se lance directement dessus.

macOS / Linux
curl -fsSL https://raw.githubusercontent.com/khalilbenaz/claude-skills-collection/main/install.sh | sh -s -- agent-mcp-server-builder --launch
Windows (PowerShell)
iex "& { $(iwr -useb https://raw.githubusercontent.com/khalilbenaz/claude-skills-collection/main/install.ps1) } agent-mcp-server-builder -Launch"

🚀 Déjà installé ?

claude "/agent-mcp-server-builder"

Ou tapez /agent-mcp-server-builder dans une session Claude Code, ou décrivez simplement votre besoin — le skill se déclenche automatiquement via le skill-router.

🔑 Déclencheurs automatiques

Le skill s'active automatiquement quand votre demande contient :

MCPModel Context ProtocolMCP serverMCP toolMCP resourceserveur MCPconnecter Claude àexposer une API à Claudeclaude desktop config

📦 Installation manuelle

git clone https://github.com/khalilbenaz/claude-skills-collection.git cp -r claude-skills-collection/skills/agent-mcp-server-builder ~/.claude/skills/

Payload du plugin : skills/agent-mcp-server-builder · source éditable : agent-skills/mcp-server-builder

📖 Manuel

MCP Server Builder

Quand utiliser ce skill

Crée un serveur MCP quand l'objectif est d'exposer une capacité (API, base de données, système de fichiers, service métier) à un LLM de façon structurée. Les cas typiques :

Critères de décision : transport

ContexteTransport recommandé
Claude Desktop / Cursor / usage localstdio
Serveur réseau, multi-clientsStreamable HTTP (depuis SDK 1.x)
Compatibilité legacy SSE requiseSSE (déprécié en 2025)
Edge / Cloudflare WorkersStreamable HTTP avec fetch handler

Workflow

1. Initialisation du projet

Python (recommandé pour démarrages rapides) :

uv init my-mcp-server && cd my-mcp-server
uv add "mcp[cli]"
uv run mcp dev server.py        # hot-reload avec MCP Inspector

TypeScript (recommandé pour production Node.js) :

npx @modelcontextprotocol/create-server my-mcp-server
cd my-mcp-server && npm install
npm run build

Structure minimale Python :

my-mcp-server/
  server.py
  pyproject.toml
  .env               # secrets, jamais committé

2. Squelette serveur minimal

Python — stdio (Claude Desktop) :

from mcp.server.fastmcp import FastMCP

mcp = FastMCP("my-server")

@mcp.tool()
async def ping() -> str:
    """Vérifie que le serveur répond."""
    return "pong"

if __name__ == "__main__":
    mcp.run()          # transport stdio par défaut

TypeScript — stdio :

import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js";
import { z } from "zod";

const server = new McpServer({ name: "my-server", version: "1.0.0" });

server.tool("ping", "Vérifie que le serveur répond", {}, async () => ({
  content: [{ type: "text", text: "pong" }],
}));

const transport = new StdioServerTransport();
await server.connect(transport);

3. Définir des Tools

Règles de nommage : snake_case, verbe d'action, sans ambiguïté (search_emails > emails).

La description est lue par le LLM pour décider quand appeler l'outil — elle doit être précise et inclure les cas limites.

from pydantic import BaseModel, Field

class SearchInput(BaseModel):
    query: str = Field(description="Texte à rechercher, supporte la syntaxe Lucene")
    limit: int = Field(default=10, ge=1, le=100, description="Nombre de résultats max")
    since: str | None = Field(default=None, description="Filtre date ISO 8601, ex: 2026-01-01")

@mcp.tool()
async def search_documents(params: SearchInput) -> list[dict]:
    """
    Recherche dans la base documentaire interne.
    Retourne titre, extrait, url et score de pertinence.
    Utiliser quand l'utilisateur cherche un document, une procédure ou une politique.
    """
    results = await db.search(params.query, params.limit, params.since)
    return [{"title": r.title, "snippet": r.snippet, "url": r.url, "score": r.score}
            for r in results]

4. Définir des Resources

Les resources exposent des données en lecture sans action. Le LLM peut les inclure dans son contexte.

@mcp.resource("config://app/{env}")
async def get_config(env: str) -> str:
    """Configuration de l'application pour l'environnement donné."""
    if env not in ("dev", "staging", "prod"):
        raise ValueError(f"Environnement inconnu : {env}")
    data = load_config(env)
    return json.dumps(data, indent=2)

URI schemes utiles : file://, db://, api://, config://, doc://.

5. Définir des Prompts

@mcp.prompt()
def code_review_prompt(language: str, code: str) -> list[dict]:
    """Template de revue de code réutilisable."""
    return [
        {"role": "user", "content": f"Revois ce code {language} :\n\n```{language}\n{code}\n```\nFocus : sécurité, perf, lisibilité."}
    ]

6. Transport réseau (Streamable HTTP)

from mcp.server.fastmcp import FastMCP
import uvicorn

mcp = FastMCP("my-server")
# ... définir tools/resources ...

app = mcp.streamable_http_app()   # renvoie une ASGI app

if __name__ == "__main__":
    uvicorn.run(app, host="0.0.0.0", port=8000)

URL cliente : http://localhost:8000/mcp

7. Error handling

from mcp.server.fastmcp import Context
from mcp.types import McpError, ErrorCode

@mcp.tool()
async def get_user(user_id: str, ctx: Context) -> dict:
    try:
        user = await db.get_user(user_id)
        if not user:
            raise McpError(ErrorCode.NOT_FOUND, f"Utilisateur {user_id} introuvable")
        return user.dict()
    except DbConnectionError as e:
        await ctx.error(f"DB indisponible : {e}")
        raise McpError(ErrorCode.INTERNAL_ERROR, "Service temporairement indisponible")

Codes d'erreur MCP : INVALID_PARAMS, NOT_FOUND, INTERNAL_ERROR, METHOD_NOT_FOUND.

8. Testing

# Inspecter interactivement avec MCP Inspector
npx @modelcontextprotocol/inspector python server.py

# Tests unitaires (Python)
uv add --dev pytest pytest-asyncio
import pytest
from mcp.client.session import ClientSession
from mcp.client.stdio import StdioServerParameters, stdio_client

@pytest.mark.asyncio
async def test_ping():
    params = StdioServerParameters(command="python", args=["server.py"])
    async with stdio_client(params) as (r, w):
        async with ClientSession(r, w) as session:
            await session.initialize()
            result = await session.call_tool("ping", {})
            assert result.content[0].text == "pong"

9. Configuration client

Claude Desktop (~/Library/Application Support/Claude/claude_desktop_config.json sur macOS) :

{
  "mcpServers": {
    "my-server": {
      "command": "uv",
      "args": ["run", "--directory", "/abs/path/my-mcp-server", "python", "server.py"],
      "env": {
        "API_KEY": "sk-..."
      }
    }
  }
}

Cursor (.cursor/mcp.json à la racine du projet) :

{
  "mcpServers": {
    "my-server": {
      "command": "node",
      "args": ["dist/index.js"]
    }
  }
}

Redémarrer le client après modification de la config.

10. Dockerfile pour déploiement réseau

FROM python:3.12-slim
WORKDIR /app
COPY pyproject.toml .
RUN pip install uv && uv sync --no-dev
COPY . .
EXPOSE 8000
CMD ["uv", "run", "python", "server.py"]

Garde-fous / Anti-patterns

Secrets dans les descriptions — Ne jamais inclure de clé API, token ou donnée sensible dans description d'un outil (le LLM peut les loguer ou les retourner dans ses réponses). Passer les secrets via env dans la config client.

Outils trop génériques — Un outil execute_sql(query: str) sans validation est dangereux. Préférer des outils spécialisés (list_users, get_order_by_id) avec validation Pydantic stricte.

Absence de timeout — Les handlers stdio bloquants font geler Claude Desktop. Toujours utiliser asyncio.wait_for avec timeout.

import asyncio

@mcp.tool()
async def slow_api_call(params: MyInput) -> dict:
    try:
        return await asyncio.wait_for(external_api(params), timeout=15.0)
    except asyncio.TimeoutError:
        raise McpError(ErrorCode.INTERNAL_ERROR, "Timeout API après 15s")

Retours non-sérialisables — Tout retour doit être JSON-sérialisable. Convertir les objets métier (datetime, Decimal, custom classes) avant de retourner.

Mélanger transport stdio et logs sur stdout — En stdio, stdout est le canal MCP. Les logs doivent aller sur stderr ou dans un fichier.

import logging, sys
logging.basicConfig(stream=sys.stderr, level=logging.INFO)

URI de ressource non-canoniques — Les templates d'URI (users/{id}) doivent correspondre exactement au pattern déclaré. Un mismatch rend la ressource silencieusement inaccessible.

Bonnes pratiques 2026