0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

FastAPI-MCPサーバーからツール情報を取得する方法

0
Posted at

はじめに

この記事では、FastAPIで構築したサーバーをMCP (Model Context Protocol) サーバー化し、クライアントからツールの情報を取得する具体的な手順を紹介します。

💡 MCPとは?
MCPは、生成AIモデルが外部のデータソースやツールと連携するための標準プロトコルです。これにより、AIモデルは文脈情報を効率的に管理し、その能力を拡張できます。いわば、AIと外部機能をつなぐための「共通言語」のようなものです。

✅ 対象読者

  • FastAPIを使ったことがあり、アプリケーションをMCPサーバー化したい方
  • MCPクライアントがサーバーから情報を取得する具体的な通信フローを知りたい方

✅ 前提知識

  • FastAPIの基礎的な知識
  • MCPの基本的な概念に関する知識

🚀 MCPサーバーの構築

MCPサーバーは、FastAPI-MCPライブラリを使って驚くほど簡単に構築できます。
既存のFastAPIアプリケーションにFastApiMCP(app).mount_http()の1行を追加するだけで、MCPサーバーとして機能させることが可能です。

🔗 公式ドキュメント

以下の内容でserver.pyを作成してください。

from fastapi import FastAPI
from fastapi_mcp import FastApiMCP

app = FastAPI()

@app.get("/hello", operation_id="hello_get", status_code=201)
def hello(user: str) -> str:
    """挨拶を返す

    :param user: 相手
    :return: 挨拶
    """
    return f"Hi {user}!"

FastApiMCP(app).mount_http()

ワンポイントアドバイス
operation_id="hello_get"を指定することで、MCPにおけるツール名がhello_getのように意図した名前になります。これを指定しない場合、FastAPIが自動生成する名前(この例ではhello_hello_get)が使われます。デバッグや管理のしやすさから、明示的に指定することをおすすめします

サーバーの起動

ターミナルで以下のコマンドを実行し、MCPサーバーを起動します。

uv run --with fastapi-mcp uvicorn server:app

補足: uvuvicorn について

  • uv: Ruffプロジェクトが開発した、高速なPythonパッケージインストーラー兼仮想環境マネージャーです。
  • uvicorn: FastAPIなどのASGIアプリケーションを動作させるための、軽量で高速なサーバーです。
    このコマンドは「uvを使ってfastapi-mcpを含む依存関係を解決し、uvicornserver.py内のappを起動する」という意味になります。

ポートの競合に注意!
デフォルトではポート8000を使用します。もしaddress already in useといったエラーが表示された場合は、他のプロセスがそのポートをすでに使用しています。
その際は、--port 8100のように空いているポート番号を指定してください。後述するクライアント側のMCP_URLも忘れずに変更しましょう。


🔧 クライアントからツール情報を取得する

サーバーからツール情報を取得するには、JSON-RPC 2.0 というプロトコルに則って通信を行います。

💡 JSON-RPC 2.0とは?
軽量なリモートプロシージャコール(RPC)プロトコルの1つです。HTTPリクエストのボディにJSON形式で「どのメソッドを (method)」「どんな引数で (params)」呼び出したいかを記述して送信します。サーバーからの応答もJSONで返され、成功時にはresult、失敗時にはerrorキーが含まれます。

情報の取得は、以下の3ステップの通信を順番に行うのが基本です。

  1. セッションの初期化 (initialize)
    • 目的: クライアントとサーバー間で「握手」を交わし、通信を開始します。このリクエストにより、この後の一連の通信で使うセッションIDが発行されます。
  2. 初期化完了の通知 (notifications/initialized)
    • 目的: クライアント側の準備が整ったことをサーバーに伝えます。これにより、サーバーはクライアントから次のリクエストを受け付ける準備ができたと認識します。
  3. ツール一覧の取得 (tools/list)
    • 目的: サーバーが提供している利用可能なツールの一覧を問い合わせます。

以下の内容でclient.pyを作成してください。

from pprint import pprint
import httpx

MCP_URL = "http://127.0.0.1:8000/mcp"

# 各リクエストのペイロードを定義
init_params = {
    "protocolVersion": "1.0",
    "capabilities": {},
    "clientInfo": {"name": "init client", "version": "1.0"},
}
headers = {"Accept": "application/json, text/event-stream"}
payload1 = {"jsonrpc": "2.0", "id": 1, "method": "initialize", "params": init_params}
payload2 = {"jsonrpc": "2.0", "method": "notifications/initialized"}
payload3 = {"jsonrpc": "2.0", "id": 3, "method": "tools/list"}

with httpx.Client() as client:
    # 1. セッションIDの取得
    init_resp = client.post(MCP_URL, headers=headers, json=payload1)
    session_id = init_resp.headers.get("Mcp-Session-Id")
    print("セッションID:", session_id)
    pprint(init_resp.json())

    # セッションIDを付与したヘッダ
    headers_with_id = headers | {"Mcp-Session-Id": session_id}

    # 2. 完了通知
    client.post(MCP_URL, headers=headers_with_id, json=payload2)

    # 3. ツール一覧の取得
    tools_resp = client.post(MCP_URL, headers=headers_with_id, json=payload3)
    tools_json = tools_resp.json()

# 結果を見やすく整形して表示
# descriptionは長いため、変数に保持して別途表示する
description = tools_json["result"]["tools"][0]["description"]
tools_json["result"]["tools"][0]["description"] = "..."
pprint(tools_json)
print("DESCRIPTION")
print(description)

プログラムの実行

サーバーを起動したまま、別のターミナルで以下のコマンドを実行します。

uv run --with httpx client.py

実行結果の例

セッションID: 01234567890123456789012345678901
{'id': 1,
 'jsonrpc': '2.0',
 'result': {'capabilities': {'experimental': {},
                             'tools': {'listChanged': False}},
            'protocolVersion': '2025-06-18',
            'serverInfo': {'name': 'FastAPI', 'version': '1.14.0'}}}
{'id': 3,
 'jsonrpc': '2.0',
 'result': {'tools': [{'description': '...',
                       'inputSchema': {'properties': {'user': {'title': 'user',
                                                               'type': 'string'}},
                                       'required': ['user'],
                                       'title': 'hello_getArguments',
                                       'type': 'object'},
                       'name': 'hello_get'}]}}
DESCRIPTION
Hello

挨拶を返す

:param user: 相手
:return: 挨拶

### Responses:

**201**: Successful Response (Success Response)
Content-Type: application/json

**Example Response:**
```json
"Response Hello Get"
```

実行結果から、FastAPIのエンドポイント(hello関数)のdocstringが、ツールのdescriptionとして正しく取得できていることが確認できます。これにより、クライアント(やAIモデル)はツールの機能や使い方を理解できます。


📚 補足情報

HTTP Transport と SSE Transport

FastAPI-MCPには、MCPの通信方式(Transport)としてHTTP TransportSSE (Server-Sent Events) Transportの2種類が用意されています。

🔗 詳細:

今回はリクエストごとに応答が返ってくるシンプルなHTTP Transportmount_http())を使用しました。サーバーからのプッシュ通知など、より双方向な通信が必要な場合はSSE Transportmount_sse())を検討しますが、その場合クライアント側の情報の取得方法も異なるため注意が必要です。

よくあるハマりポイント

  • セッションIDの付け忘れ
    • initialize後のnotifications/initializedtools/listといったリクエストでは、必ずHTTPヘッダーに取得したセッションID (Mcp-Session-Id) を含める必要があります。忘れるとサーバー側でセッションを特定できずエラーになります。
  • ポートの競合
    • 前述の通り、uvicorn起動時にポートが使用済みの場合があります。--portオプションで空いているポートを指定しましょう。
  • Acceptヘッダーの指定
    • クライアントからリクエストを送る際、ヘッダーに"Accept": "application/json, text/event-stream"を指定しないと、サーバーが応答形式を決定できず406 Not Acceptableエラーが発生することがあります。

🎉 まとめ

本記事では、FastAPI-MCPを使ってMCPサーバーを構築し、クライアントからツール情報を取得するまでの一連の流れを解説しました。

  • サーバー構築は簡単: FastAPI-MCPを使えば、既存のFastAPIアプリに1行追加するだけでMCPサーバー化できます。
  • 通信プロトコル: クライアントとの通信には、軽量なJSON-RPC 2.0プロトコルが使われます。
  • 通信の順序が重要: initializenotifications/initializedtools/list の順番でリクエストを送るのが基本フローです。

この手順を理解することで、生成AIと連携可能なカスタムツールを効率的に開発する第一歩となります。

0
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?