![[備忘録] Gradio MCPサーバーを試してみたら、docstringの重要性に気づいた話 #Python - Qiita](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-user-contents.imgix.net%2Fhttps%253A%252F%252Fcdn.qiita.com%252Fassets%252Fpublic%252Farticle-ogp-background-afbab5eb44e0b055cce1258705637a91.png%3Fixlib%3Drb-4.0.0%26w%3D1200%26blend64%3DaHR0cHM6Ly9xaWl0YS11c2VyLXByb2ZpbGUtaW1hZ2VzLmltZ2l4Lm5ldC9odHRwcyUzQSUyRiUyRnMzLWFwLW5vcnRoZWFzdC0xLmFtYXpvbmF3cy5jb20lMkZxaWl0YS1pbWFnZS1zdG9yZSUyRjAlMkYyNjQ4MDY5JTJGMGU0YTU4OTYzNzI5ZTIyYjkwODM5NDY4YjkzYWNkMGNmNWM4YWFhOCUyRmxhcmdlLnBuZyUzRjE2NTE1ODc2MzQ_aXhsaWI9cmItNC4wLjAmYXI9MSUzQTEmZml0PWNyb3AmbWFzaz1lbGxpcHNlJmZtPXBuZzMyJnM9ZmQxZmIxNGYyM2JiYTE1NTZmMzQ4YmYxZGE3YzFmMjc%26blend-x%3D120%26blend-y%3D467%26blend-w%3D82%26blend-h%3D82%26blend-mode%3Dnormal%26s%3D4e21d6c99ffd21130be46a8d859f162c?ixlib=rb-4.0.0&w=1200&fm=jpg&mark64=aHR0cHM6Ly9xaWl0YS11c2VyLWNvbnRlbnRzLmltZ2l4Lm5ldC9-dGV4dD9peGxpYj1yYi00LjAuMCZ3PTk2MCZoPTMyNCZ0eHQ9JTVCJUU1JTgyJTk5JUU1JUJGJTk4JUU5JThDJUIyJTVEJTIwR3JhZGlvJTIwTUNQJUUzJTgyJUI1JUUzJTgzJUJDJUUzJTgzJTkwJUUzJTgzJUJDJUUzJTgyJTkyJUU4JUE5JUE2JUUzJTgxJTk3JUUzJTgxJUE2JUUzJTgxJUJGJUUzJTgxJTlGJUUzJTgyJTg5JUUzJTgwJTgxZG9jc3RyaW5nJUUzJTgxJUFFJUU5JTg3JThEJUU4JUE2JTgxJUU2JTgwJUE3JUUzJTgxJUFCJUU2JUIwJTk3JUUzJTgxJUE1JUUzJTgxJTg0JUUzJTgxJTlGJUU4JUE5JUIxJnR4dC1hbGlnbj1sZWZ0JTJDdG9wJnR4dC1jb2xvcj0lMjMxRTIxMjEmdHh0LWZvbnQ9SGlyYWdpbm8lMjBTYW5zJTIwVzYmdHh0LXNpemU9NTYmdHh0LXBhZD0wJnM9Y2EyMmIxNTk4NmYzOGJhMjk4MzAzNjdhNzA2N2QwZWY&mark-x=120&mark-y=112&blend64=aHR0cHM6Ly9xaWl0YS11c2VyLWNvbnRlbnRzLmltZ2l4Lm5ldC9-dGV4dD9peGxpYj1yYi00LjAuMCZ3PTgzOCZoPTU4JnR4dD0lNDBUYWRhdGFrYV9UYWthaGFzaGkmdHh0LWNvbG9yPSUyMzFFMjEyMSZ0eHQtZm9udD1IaXJhZ2lubyUyMFNhbnMlMjBXNiZ0eHQtc2l6ZT0zNiZ0eHQtcGFkPTAmcz01YWM4MmFhMTM4Y2ZkZDI1MzAyNzNkMmI1YzJiNDBhMQ&blend-x=242&blend-y=480&blend-w=838&blend-h=46&blend-fit=crop&blend-crop=left%2Cbottom&blend-mode=normal&s=c7123a6c98ad90a454fd624f5c43355f)
こんにちは!最近、Gradioのmcp_server=Trueという機能を試してみて驚いたことがあります。1行の設定で、Python関数がClaudeなどのLLMと連携できるようになります。でも、なぜこれが可能なのか?その秘密を探るうちに、意外な主役が見つかりました—それがdocstring(関数のコメント)だったのです。
はじめに:1行の設定なのか (?)
Gradioは、Pythonでチャットボット、データ分析のUIを簡単に作れるライブラリとして知られています。最近のバージョンでは、MCPサーバー機能が追加されました。
demo.launch(mcp_server=True)
1行で、あなたのGradioアプリはMCP(Model Context Protocol)サーバーとして動作し、Claudeなどの大規模言語モデルから直接呼び出せるようになります。でも、この機能の裏側では、何が起きているのでしょうか?
MCPとは何か?簡単に説明
MCP(Model Control Protocol)は、大規模言語モデル(LLM)が外部ツールや関数を自然言語ベースで呼び出すための軽量なプロトコルであり、計算処理や画像生成などを外部リソースに委ねることを可能にします。
実際に試してみた
つぎのページの公式のページの最初のサンプルプログラムを試してみました。
import gradio as gr
def letter_counter(word, letter):
"""
Count the number of occurrences of a letter in a word or text.
Args:
word (str): The input text to search through
letter (str): The letter to search for
Returns:
int: The number of times the letter appears
"""
word = word.lower()
letter = letter.lower()
count = word.count(letter)
return count
demo = gr.Interface(
fn=letter_counter,
inputs=["textbox", "textbox"],
outputs="number",
title="Letter Counter",
description="Enter text and a letter to count how many times the letter appears in the text."
)
if __name__ == "__main__":
demo.launch(mcp_server=True) # MCPサーバーとして起動
このアプリを起動すると、コンソールにMCPサーバーのURLが表示されます:
gradioのUIのページに遷移すると以下のように、wordのletterをカウントしてくれるアプリが起動します。
このURLをMCPクライアントの設定に追加すると…驚くべきことに、MCPサーバー化して連携できるようです。以下のように自分の場合はvscodeの設定のmcpの設定 (gradio01の項目) を追記して、サーバーを起動して試してみました。
"mcp": {
"inputs": [],
"servers": {
"mcp-server-time": {
"command": "python",
"args": [
"-m",
"mcp_server_time",
"--local-timezone=America/Los_Angeles"
],
"env": {}
},
"gradio01': {
"type": "sse",
"url": "http://127.0.0.1:7860/gradio_api/mcp/sse"
},
}
}
},
設定の再生ボタンから起動すると、上記のように起動します。
GitHub Copilotのチャットの画面を起動して、エージェントモードを選択します。
ツールのアイコンをクリックすると、どのmcpサーバと連携するか一覧がでてくるので、対象のmcpサーバを選択します。
「strawberryの中にrはいくつありますか?」と質問すると、文字カウント関数を使って正確に答えてくれました。
続行を選択する。
なぜこれが動くのか?秘密はdocstringにあった
なぜLLMはgradioのUIや関数の入出力について理解できたのでしょうか?
実装を調べてみると、その答えにたどり着きました。
Gradioの内部実装(以下はGradioのソースコードの一部)を見ると、get_function_descriptionという関数が呼び出されています:
description, parameters = utils.get_function_description(block_fn.fn)
この関数は何をしているのでしょうか?実装を見てみると:
def get_function_description(fn: Callable) -> tuple[str, dict[str, str]]:
"""
Get the description of a function and its parameters by parsing the docstring.
...
"""
fn_docstring = fn.__doc__
description = ""
parameters = {}
if not fn_docstring:
return description, parameters
lines = fn_docstring.strip().split("\n")
# 説明部分の抽出
description_lines = []
for line in lines:
line = line.strip()
if line.startswith(("Args:", "Parameters:", "Arguments:")):
break
if line:
description_lines.append(line)
description = " ".join(description_lines)
# パラメータ部分の抽出
try:
param_start_idx = next(
(
i
for i, line in enumerate(lines)
if line.strip().startswith(("Args:", "Parameters:", "Arguments:"))
),
len(lines),
)
for line in lines[param_start_idx + 1 :]:
line = line.strip()
if not line:
continue
try:
if ":" in line:
param_name, param_desc = line.split(":", 1)
param_name = param_name.split(" ")[0].strip()
if param_name:
parameters[param_name] = param_desc.strip()
except Exception:
continue
except Exception:
pass
return description, parameters
そうです!Gradioは関数のdocstring(コメント)を解析して、その内容をMCPツールの説明として使っているのです。関数の説明とパラメータの説明がdocstringから抽出され、LLMに提供されています。
これが、Docstringの書き方が重要である理由です。gr.Interface に渡す title や description、そして docstring は、単なるコメントではなく、大規模言語モデル(LLM)と連携する際の重要なメタ情報として利用されます。
効果的なdocstringの4つの要素
MCPツールとして機能するdocstringには以下の要素が重要です:
- 関数の明確な説明 – 最初に何をする関数かを簡潔に
-
パラメータセクションの明示 –
Args:などの見出しを使用 - パラメータの詳細説明 – 型情報と役割を明記
-
戻り値の説明 –
Returns:セクションを追加
これらの要素を含めることで、LLMがあなたの関数を理解し、適切に活用できるようになります。
まとめ:docstringはLLM連携の鍵
Gradio MCPサーバーの驚くべき点は、普段何気なく書いているdocstringが、LLMとの連携の要になることです。適切に書かれたdocstringは、そのままLLMへの説明書として機能し、関数の目的とパラメータを伝えています。
つまり、docstringの質がLLM連携の質を左右します。明確で構造化されたdocstringを意識して書くことが、大事だということがわかりました。
参考リンク
docstringなしで起動した場合
以下のような警告が出力される
免責事項
- 本記事の作成にあたり、文章および図解の一部に生成AIを活用しています。最終的な編集および内容の確認は筆者が責任を持って行っています。
- 本記事は2025年5月初旬時点の情報に基づいて記載しています。仕様や実装内容は将来的に変更される可能性があるため、最新の情報は必ず公式サイト等でご確認ください。
Views: 0











