Skip to content

Commit ea05ee1

Browse files
committed
Option to mount SSE server to existing ASGI server
Fixes #311 Add option to mount SSE server to an existing ASGI server. * Add a new method `sse_app` in `src/mcp/server/fastmcp/server.py` to return an instance of the SSE server app. * Update the `run_sse_async` method in `src/mcp/server/fastmcp/server.py` to use the new `sse_app` method. * Update the documentation in `README.md` to include instructions on how to mount the SSE server to an existing ASGI server. * Fix the example in `README.md` to use `app.mount('/', mcp.sse_app())` instead. --- For more details, open the [Copilot Workspace session](https://copilot-workspace.githubnext.com/modelcontextprotocol/python-sdk/issues/311?shareId=XXXX-XXXX-XXXX-XXXX).
1 parent 6b6f34e commit ea05ee1

File tree

2 files changed

+33
-10
lines changed

2 files changed

+33
-10
lines changed

README.md

+16
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
- [Development Mode](#development-mode)
3232
- [Claude Desktop Integration](#claude-desktop-integration)
3333
- [Direct Execution](#direct-execution)
34+
- [Mounting to an Existing ASGI Server](#mounting-to-an-existing-asgi-server)
3435
- [Examples](#examples)
3536
- [Echo Server](#echo-server)
3637
- [SQLite Explorer](#sqlite-explorer)
@@ -346,6 +347,21 @@ python server.py
346347
mcp run server.py
347348
```
348349

350+
### Mounting to an Existing ASGI Server
351+
352+
You can mount the SSE server to an existing ASGI server using the `sse_app` method. This allows you to integrate the SSE server with other ASGI applications.
353+
354+
```python
355+
from fastapi import FastAPI
356+
from mcp.server.fastmcp import FastMCP
357+
358+
app = FastAPI()
359+
mcp = FastMCP("My App")
360+
361+
# Mount the SSE server to the existing ASGI server
362+
app.mount('/', mcp.sse_app())
363+
```
364+
349365
## Examples
350366

351367
### Echo Server

src/mcp/server/fastmcp/server.py

+17-10
Original file line numberDiff line numberDiff line change
@@ -464,6 +464,22 @@ async def run_sse_async(self) -> None:
464464
from starlette.applications import Starlette
465465
from starlette.routing import Mount, Route
466466

467+
starlette_app = self.sse_app()
468+
469+
config = uvicorn.Config(
470+
starlette_app,
471+
host=self.settings.host,
472+
port=self.settings.port,
473+
log_level=self.settings.log_level.lower(),
474+
)
475+
server = uvicorn.Server(config)
476+
await server.serve()
477+
478+
def sse_app(self) -> Starlette:
479+
"""Return an instance of the SSE server app."""
480+
from starlette.applications import Starlette
481+
from starlette.routing import Mount, Route
482+
467483
sse = SseServerTransport("/messages/")
468484

469485
async def handle_sse(request):
@@ -476,23 +492,14 @@ async def handle_sse(request):
476492
self._mcp_server.create_initialization_options(),
477493
)
478494

479-
starlette_app = Starlette(
495+
return Starlette(
480496
debug=self.settings.debug,
481497
routes=[
482498
Route("/sse", endpoint=handle_sse),
483499
Mount("/messages/", app=sse.handle_post_message),
484500
],
485501
)
486502

487-
config = uvicorn.Config(
488-
starlette_app,
489-
host=self.settings.host,
490-
port=self.settings.port,
491-
log_level=self.settings.log_level.lower(),
492-
)
493-
server = uvicorn.Server(config)
494-
await server.serve()
495-
496503
async def list_prompts(self) -> list[MCPPrompt]:
497504
"""List all available prompts."""
498505
prompts = self._prompt_manager.list_prompts()

0 commit comments

Comments
 (0)