Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: Watch startup errors for stdio client #289

Open
wants to merge 12 commits into
base: main
Choose a base branch
from
Open
158 changes: 158 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,158 @@
# Changelog

## [1.3.0] - 2025-02-20

### Breaking Changes

- **Context API Changes**: The Context logging methods (info, debug, warning, error) are now async and must be awaited. ([#172](https://github.com/modelcontextprotocol/python-sdk/pull/172))
- **Resource Response Format**: Standardized resource response format to return both content and MIME type. Method `read_resource()` now returns a tuple of `(content, mime_type)` instead of just content. ([#170](https://github.com/modelcontextprotocol/python-sdk/pull/170))

### New Features

#### Lifespan Support
Added comprehensive server lifecycle management through the lifespan API:
```python
@dataclass
class AppContext:
db: Database

@asynccontextmanager
async def app_lifespan(server: FastMCP) -> AsyncIterator[AppContext]:
try:
await db.connect()
yield AppContext(db=db)
finally:
await db.disconnect()

mcp = FastMCP("My App", lifespan=app_lifespan)

@mcp.tool()
def query_db(ctx: Context) -> str:
db = ctx.request_context.lifespan_context["db"]
return db.query()
```
([#203](https://github.com/modelcontextprotocol/python-sdk/pull/203))

#### Async Resources
Added support for async resource functions in FastMCP:
```python
@mcp.resource("users://{user_id}")
async def get_user(user_id: str) -> str:
async with client.session() as session:
response = await session.get(f"/users/{user_id}")
return await response.text()
```
([#157](https://github.com/modelcontextprotocol/python-sdk/pull/157))

#### Concurrent Request Handling
Made message handling concurrent, allowing multiple requests to be processed simultaneously. ([#206](https://github.com/modelcontextprotocol/python-sdk/pull/206))

#### Request Cancellation
Added support for canceling in-flight requests and cleaning up resources. ([#167](https://github.com/modelcontextprotocol/python-sdk/pull/167))

#### Server Instructions
Added support for the `instructions` field in server initialization, allowing servers to provide usage guidance. ([#150](https://github.com/modelcontextprotocol/python-sdk/pull/150))

### Bug Fixes

- Fixed progress reporting for first tool call by correcting progress_token handling ([#176](https://github.com/modelcontextprotocol/python-sdk/pull/176))
- Fixed server crash when using debug logging ([#158](https://github.com/modelcontextprotocol/python-sdk/pull/158))
- Fixed resource template handling in FastMCP server ([#137](https://github.com/modelcontextprotocol/python-sdk/pull/137))
- Fixed MIME type preservation in resource responses ([#170](https://github.com/modelcontextprotocol/python-sdk/pull/170))
- Fixed documentation for environment variables in CLI commands ([#149](https://github.com/modelcontextprotocol/python-sdk/pull/149))
- Fixed request ID preservation in JSON-RPC responses ([#205](https://github.com/modelcontextprotocol/python-sdk/pull/205))

### Dependency Updates

- Relaxed version constraints for better compatibility:
- `pydantic`: Changed from `>=2.10.1,<3.0.0` to `>=2.7.2,<3.0.0`
- `pydantic-settings`: Changed from `>=2.6.1` to `>=2.5.2`
- `uvicorn`: Changed from `>=0.30` to `>=0.23.1`
([#180](https://github.com/modelcontextprotocol/python-sdk/pull/180))

### Examples

- Added a simple chatbot example client to demonstrate SDK usage ([#98](https://github.com/modelcontextprotocol/python-sdk/pull/98))

### Client Improvements

- Added client support for sampling, list roots, and ping requests ([#218](https://github.com/modelcontextprotocol/python-sdk/pull/218))
- Added flexible type system for tool result returns ([#222](https://github.com/modelcontextprotocol/python-sdk/pull/222))

### Compatibility and Platform Support

- Updated URL validation to allow file and other nonstandard schemas ([#68fcf92](https://github.com/modelcontextprotocol/python-sdk/commit/68fcf92947f7d02d50340053a72a969d6bb70e1b))
- Force stdin/stdout encoding to UTF-8 for cross-platform compatibility ([#d92ee8f](https://github.com/modelcontextprotocol/python-sdk/commit/d92ee8feaa5675efddd399f3e8ebe8ed976b84c2))

### Internal Improvements

- Improved type annotations for better IDE support ([#181](https://github.com/modelcontextprotocol/python-sdk/pull/181))
- Added comprehensive tests for SSE transport ([#151](https://github.com/modelcontextprotocol/python-sdk/pull/151))
- Updated types to match 2024-11-05 MCP schema ([#165](https://github.com/modelcontextprotocol/python-sdk/pull/165))
- Refactored request and notification handling for better code organization ([#166](https://github.com/modelcontextprotocol/python-sdk/pull/166))

## [1.2.1] - 2024-01-27

### Added
- Support for async resources
- Example and test for parameter descriptions in FastMCP tools

### Fixed
- MCP install command with environment variables
- Resource template handling in FastMCP server (#129)
- Package in the generated MCP run config (#128)
- FastMCP logger debug output
- Handling of strings containing numbers in FastMCP (@sd2k, #142)

### Changed
- Refactored to use built-in typing.Annotated instead of typing_extensions
- Updated uv.lock
- Added .DS_Store to gitignore

# MCP Python SDK v1.2.0rc1 Release Notes

## Major Features

### FastMCP Integration
- Integrated [FastMCP](https://github.com/jlowin/fastmcp) as the recommended high-level server framework
- Added new `mcp.server.fastmcp` module with simplified decorator-based API
- Introduced `FastMCP` class for easier server creation and management
- Added comprehensive documentation and examples for FastMCP usage

### New CLI Package
- Added new CLI package for improved developer experience
- Introduced `mcp dev` command for local development and testing
- Added `mcp install` command for Claude Desktop integration
- Added `mcp run` command for direct server execution

## Improvements

### Documentation
- Completely revamped README with new structure and examples
- Added detailed sections on core concepts (Resources, Tools, Prompts)
- Updated documentation to recommend FastMCP as primary API
- Added sections on development workflow and deployment options
- Improved example server documentation

### Developer Experience
- Added pre-commit hooks for code quality
- Updated to Pydantic 2.10.0 for improved type checking
- Added uvicorn as a dependency for better server capabilities

## Bug Fixes
- Fixed deprecation warnings in core components
- Fixed Pydantic field handling for meta fields
- Fixed type issues throughout the codebase
- Fixed example server READMEs

## Breaking Changes
- Deprecated direct usage of `mcp.server` in favor of `mcp.server.fastmcp`
- Updated import paths for FastMCP integration
- Changed recommended installation to include CLI features (`pip install "mcp[cli]"`)

## Contributors
Special thanks to all contributors who made this release possible, including:
- Jeremiah Lowin (FastMCP)
- Oskar Raszkiewicz

**Full Changelog**: https://github.com/modelcontextprotocol/python-sdk/compare/v1.1.2...v1.2.0rc1
2 changes: 2 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -532,6 +532,7 @@ server_params = StdioServerParameters(
)



# Optional: create a sampling callback
async def handle_sampling_message(
message: types.CreateMessageRequestParams,
Expand All @@ -547,6 +548,7 @@ async def handle_sampling_message(
)



async def run():
async with stdio_client(server_params) as (read, write):
async with ClientSession(
Expand Down
1 change: 1 addition & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ dependencies = [
"sse-starlette>=1.6.1",
"pydantic-settings>=2.5.2",
"uvicorn>=0.23.1",
"exceptiongroup>=1.2.2",
]

[project.optional-dependencies]
Expand Down
1 change: 1 addition & 0 deletions src/mcp/client/session.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ async def _default_list_roots_callback(
)



ClientResponse: TypeAdapter[types.ClientResult | types.ErrorData] = TypeAdapter(
types.ClientResult | types.ErrorData
)
Expand Down
37 changes: 27 additions & 10 deletions src/mcp/client/stdio.py
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,7 @@ class StdioServerParameters(BaseModel):


@asynccontextmanager
async def stdio_client(server: StdioServerParameters):
async def stdio_client(server: StdioServerParameters, startup_wait_time: float = 0.0):
"""
Client transport for stdio: this will connect to a server by spawning a
process and communicating with it over stdin/stdout.
Expand All @@ -97,11 +97,17 @@ async def stdio_client(server: StdioServerParameters):
read_stream_writer, read_stream = anyio.create_memory_object_stream(0)
write_stream, write_stream_reader = anyio.create_memory_object_stream(0)

process = await anyio.open_process(
[server.command, *server.args],
env=server.env if server.env is not None else get_default_environment(),
stderr=sys.stderr,
)
try:
process = await anyio.open_process(
[server.command, *server.args],
env=server.env or get_default_environment(),
stderr=sys.stderr,
)
except OSError as exc:
raise RuntimeError(
f"Failed to spawn process: {server.command} {server.args}. "
f"Check that the binary exists and is executable."
) from exc

async def stdout_reader():
assert process.stdout, "Opened process is missing stdout"
Expand Down Expand Up @@ -144,10 +150,21 @@ async def stdin_writer():
except anyio.ClosedResourceError:
await anyio.lowlevel.checkpoint()

async with (
anyio.create_task_group() as tg,
process,
):
async def watch_process_exit():
returncode = await process.wait()
if returncode != 0:
raise RuntimeError(
f"Subprocess exited with code {returncode}. "
f"Command: {server.command}, {server.args}"
)

async with anyio.create_task_group() as tg, process:
tg.start_soon(stdout_reader)
tg.start_soon(stdin_writer)
tg.start_soon(watch_process_exit)

if startup_wait_time > 0:
with anyio.move_on_after(startup_wait_time):
await anyio.sleep_forever()

yield read_stream, write_stream
42 changes: 33 additions & 9 deletions tests/client/test_stdio.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,11 @@
import re
import shutil
import sys

if sys.version_info < (3, 11):
from exceptiongroup import ExceptionGroup
else:
ExceptionGroup = ExceptionGroup

import pytest

Expand All @@ -14,7 +21,6 @@ async def test_stdio_client():
server_parameters = StdioServerParameters(command=tee)

async with stdio_client(server_parameters) as (read_stream, write_stream):
# Test sending and receiving messages
messages = [
JSONRPCMessage(root=JSONRPCRequest(jsonrpc="2.0", id=1, method="ping")),
JSONRPCMessage(root=JSONRPCResponse(jsonrpc="2.0", id=2, result={})),
Expand All @@ -31,13 +37,31 @@ async def test_stdio_client():
raise message

read_messages.append(message)
if len(read_messages) == 2:
if len(read_messages) == len(messages):
break

assert len(read_messages) == 2
assert read_messages[0] == JSONRPCMessage(
root=JSONRPCRequest(jsonrpc="2.0", id=1, method="ping")
)
assert read_messages[1] == JSONRPCMessage(
root=JSONRPCResponse(jsonrpc="2.0", id=2, result={})
)
assert read_messages == messages


@pytest.mark.anyio
async def test_stdio_client_spawn_failure():
server_parameters = StdioServerParameters(command="/does/not/exist")

with pytest.raises(RuntimeError, match="Failed to spawn process"):
async with stdio_client(server_parameters):
pytest.fail("Should never be reached.")


@pytest.mark.anyio
async def test_stdio_client_nonzero_exit():
server_parameters = StdioServerParameters(
command="python", args=["-c", "import sys; sys.exit(2)"]
)

with pytest.raises(ExceptionGroup) as eg_info:
async with stdio_client(server_parameters, startup_wait_time=0.2):
pytest.fail("Should never be reached.")

exc = eg_info.value.exceptions[0]
assert isinstance(exc, RuntimeError)
assert re.search(r"exited with code 2", str(exc))
3 changes: 3 additions & 0 deletions uv.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading