Flask-MCP-Server is a lightweight, production-ready Model Context Protocol (MCP) server for Flask, with tool discovery, endpoint registration, async support, and full compliance with the latest MCP 2025-06-18 specification.
Project description
flask-mcp-server
Flask-based Model Context Protocol (MCP) server for Python. Drop it into any Flask app or run it standalone. Ships with security and ops features out of the box.
Spec target: MCP 2025-06-18 — Streamable HTTP transport with a unified /mcp endpoint (POST for JSON-RPC; GET for SSE). Back-compat routes retained.
Package version: 0.6.0
Table of Contents
- Requirements
- Install
- Quick Start
- What’s new in 0.6.0
- Configuration (env vars)
- Core Concepts
- Examples
- FAQ
- Troubleshooting
- License
Requirements
- Python 3.9+
- Flask 3.x (auto-installed)
Install
From PyPI
pip install flask-mcp-server
Add to requirements.txt from a file path:
flask-mcp-server==0.6.0
Quick Start
Integrated into an existing Flask app (integrated HTTP):
from flask import Flask
from flask_mcp_server import mount_mcp, Mcp
from flask_mcp_server.http_integrated import mw_auth, mw_ratelimit, mw_cors
app = Flask(__name__)
@Mcp.tool(name="sum")
def sum_(a: int, b: int) -> int:
return a + b
# Mount at /mcp with useful middlewares
mount_mcp(app, url_prefix="/mcp", middlewares=[mw_auth, mw_ratelimit, mw_cors])
if __name__ == "__main__":
app.run(port=8765)
Dedicated server app (ships with docs endpoints):
flask-mcp serve-http
# Swagger: http://127.0.0.1:8765/swagger
# OpenAPI: /docs.json
STDIO transport:
flask-mcp serve-stdio
What’s new in 0.6.0
- Unified MCP endpoint (
/mcp):- POST a single JSON-RPC message (request/notification/response). If the
Acceptheader includestext/event-stream, the server streams the response via SSE; otherwise returns JSON. - GET to open an SSE stream for server messages (optional). Supports
Last-Event-IDfor resumability in future minor releases.
- POST a single JSON-RPC message (request/notification/response). If the
- Protocol header: Accepts
MCP-Protocol-Version: 2025-06-18(or2025-03-26for back-compat); others =>400 Bad Request. - Sessions (minimal): If the POSTed method is
"initialize", response includesMcp-Session-Idheader. Clients can include this header on subsequent requests. - Origin validation: Optional
FLASK_MCP_ALLOWED_ORIGINS(comma-separated). If set, only requests with matchingOriginare allowed at the unified MCP endpoint. - Backwards compatibility: legacy endpoints
/mcp/list,/mcp/call,/mcp/batchremain.
Configuration (env vars)
Auth & roles
export FLASK_MCP_AUTH_MODE=apikey # none|apikey|hmac
export FLASK_MCP_API_KEYS="k1,k2"
export FLASK_MCP_API_KEYS_MAP="k1:admin|user;k2:user" # per-key roles
export FLASK_MCP_HMAC_SECRET="supersecret" # if using HMAC
Rate limiting
export FLASK_MCP_RATE_LIMIT=60/m # format: <N>/<s|m|h|d>
export FLASK_MCP_RATE_SCOPE=key # ip|key
CORS, Origins & Logging
export FLASK_MCP_ALLOWED_ORIGINS="http://localhost:3000,https://your.app"
export FLASK_MCP_CORS_ORIGIN="*"
export FLASK_MCP_LOG_FORMAT=json
Providers autoload
export FLASK_MCP_PROVIDERS="my_pkg.mcp_provider:Provider,another.mod:Provider"
Core Concepts
- Registry — stores tools, resources, prompts, completions.
- Decorators —
@tool,@resource,@prompt,@completion_providerregister elements. - Facade —
Mcp.tool(...)etc. (fluent, same params as decorators). - Roles/ACL — per-element role lists checked at call time.
- TTL cache — per-tool/resource memoization via
ttl=int_seconds(memory backend). - Middleware — pluggable pipeline for integrated HTTP routes.
- Service Providers —
register(container, registry)andboot(app, registry)to wire services/routes. - Transports — Streamable HTTP unified endpoint, STDIO, and SSE notifications (hello + registry change).
Examples
A. Unified MCP endpoint usage
# List registry (JSON)
curl -s -X POST http://127.0.0.1:8765/mcp \
-H "Content-Type: application/json" \
-H "MCP-Protocol-Version: 2025-06-18" \
-d '{"jsonrpc":"2.0","id":1,"method":"mcp.list"}' | jq .
# Call a tool over SSE
curl -N -X POST http://127.0.0.1:8765/mcp \
-H "Accept: text/event-stream, application/json" \
-H "Content-Type: application/json" \
-d '{"jsonrpc":"2.0","id":"2","method":"mcp.call","params":{"kind":"tool","name":"sum","args":{"a":5,"b":7}}}'
B. Tools with roles & TTL cache
from flask_mcp_server import Mcp
@Mcp.tool(name="math.add", roles=["user","admin"], ttl=30)
def add(a: int, b: int) -> int:
return a + b
C. Resources & ResourceTemplates
from flask_mcp_server import resource, ResourceTemplate
@resource(name="profile", ttl=15)
def profile(user_id: int) -> dict:
return {"id": user_id, "name": "Alice"}
tpl = ResourceTemplate("https://api.example.com/items/{id}")
url = tpl.expand(id=42) # "https://api.example.com/items/42"
D. Prompts & Completions
from flask_mcp_server import prompt, completion_provider
from typing import List
@prompt(name="greet")
def greet(name: str) -> str:
return f"Write a warm one-line greeting for {name}."
@completion_provider(name="cities")
def cities(prefix: str="") -> List[str]:
base = ["Dhaka","Chittagong","Khulna","Rajshahi"]
return [c for c in base if c.lower().startswith(prefix.lower())]
E. Batch Calls (compat)
curl -s -X POST http://127.0.0.1:8765/mcp/batch \
-H "Content-Type: application/json" \
-d '[
{"kind":"tool","name":"math.add","args":{"a":1,"b":2}},
{"kind":"prompt","name":"greet","args":{"name":"Bashar"}}
]' | jq .
F. SSE (hello + registry events)
# GET /mcp opens an SSE stream and emits a hello event and future registry change events.
curl -N http://127.0.0.1:8765/mcp
G. Auth: API keys & HMAC
export FLASK_MCP_AUTH_MODE=apikey
export FLASK_MCP_API_KEYS="secret123"
curl -s http://127.0.0.1:8765/mcp/list -H "X-API-Key: secret123"
# HMAC
python - <<'PY'
import hmac, hashlib, json, requests
secret = b"supersecret"
body = json.dumps({"jsonrpc":"2.0","id":1,"method":"mcp.list"}).encode()
sig = hmac.new(secret, body, hashlib.sha256).hexdigest()
print(requests.post("http://127.0.0.1:8765/mcp",
headers={"X-Signature":"sha256="+sig,"Content-Type":"application/json"},
data=body).text)
PY
H. Rate limiting
export FLASK_MCP_RATE_LIMIT=100/m
export FLASK_MCP_RATE_SCOPE=key # or 'ip'
I. Service Providers
# my_pkg/mcp_provider.py
from flask_mcp_server import ServiceProvider, Mcp
class Provider(ServiceProvider):
def register(self, container, registry):
@Mcp.tool(name="time.now")
def now() -> str:
import time; return str(int(time.time()))
J. Discovery (auto-register)
# Suppose your tools live in package 'my_tools'
python -c "from flask_mcp_server.discovery import discover_package; discover_package('my_tools')"
K. CLI
flask-mcp serve-http # unified /mcp endpoint
flask-mcp serve-stdio # stdio transport
flask-mcp list # print registry
L. OpenAPI & Swagger
/docs.json— OpenAPI 3.1 JSON/swagger— Swagger UI
FAQ
Is Redis required? No. 0.6.0 uses in-memory backends by default. Redis backends will land in 0.6.x as optional extras.
Can I mount it into any Flask project? Yes—mount_mcp(app, url_prefix="/mcp").
Does it fully implement all MCP JSON-RPC methods? It implements the transport semantics and common operations (mcp.list, mcp.call, compat endpoints). If you need additional method names or shapes to match a specific client, open an issue or extend in a Service Provider.
Troubleshooting
- 400 unsupported_protocol_version — set
MCP-Protocol-Version: 2025-06-18. - 403 origin_not_allowed — configure
FLASK_MCP_ALLOWED_ORIGINSor remove it. - 401 invalid_api_key / invalid_signature — check headers and env vars.
- 429 rate_limited — adjust
FLASK_MCP_RATE_LIMITor scope.
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file flask_mcp_server-0.6.0.tar.gz.
File metadata
- Download URL: flask_mcp_server-0.6.0.tar.gz
- Upload date:
- Size: 18.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.9.6
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
aa08042f28c5d6f43c92b69e0b3c7131f0ec46c9dbd45647c9c95d335a5169d1
|
|
| MD5 |
43a680e9fd465109353a93f79f4c8773
|
|
| BLAKE2b-256 |
f4cc34b0a9bedd76116521aefc5ccf26e5e5b164df35cc2eb9fe7633644c7771
|
File details
Details for the file flask_mcp_server-0.6.0-py3-none-any.whl.
File metadata
- Download URL: flask_mcp_server-0.6.0-py3-none-any.whl
- Upload date:
- Size: 20.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.9.6
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
63d708bbf59109efa805857a5c6be5bf4a6e563ec5aaea2907ff4a59c2d1bcaa
|
|
| MD5 |
34ba04d63dcc69fcb7f00279e949d8b6
|
|
| BLAKE2b-256 |
798672b756750555c65e1c1e0bea6d53bab0c9cbb5bd6452e4a2ac5110a6192e
|