-
Notifications
You must be signed in to change notification settings - Fork 755
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
Client hangs when ctx.info() logging with pure Python client #201
Comments
Move memory stream type definitions to models.py and use them throughout the codebase for better type safety and maintainability. GitHub-Issue:#201
Updates test files to work with the ParsedMessage stream type aliases and fixes a line length issue in test_201_client_hangs_on_logging.py. Github-Issue:#201
Updates test files to work with the ParsedMessage stream type aliases and fixes a line length issue in test_201_client_hangs_on_logging.py. Github-Issue:#201
Move memory stream type definitions to models.py and use them throughout the codebase for better type safety and maintainability. GitHub-Issue:#201
Updates test files to work with the ParsedMessage stream type aliases and fixes a line length issue in test_201_client_hangs_on_logging.py. Github-Issue:#201
…ity (#239) * refactor: improve typing with memory stream type aliases Move memory stream type definitions to models.py and use them throughout the codebase for better type safety and maintainability. GitHub-Issue:#201 * refactor: move streams to ParsedMessage * refactor: update test files to use ParsedMessage Updates test files to work with the ParsedMessage stream type aliases and fixes a line length issue in test_201_client_hangs_on_logging.py. Github-Issue:#201 * refactor: rename ParsedMessage to MessageFrame for clarity 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <[email protected]> * refactor: move MessageFrame class to types.py for better code organization 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <[email protected]> * fix pyright * refactor: update websocket client to use MessageFrame Modified the websocket client to work with the new MessageFrame type, preserving raw message text and properly extracting the root JSON-RPC message when sending. Github-Issue:#204 * fix: use NoneType instead of None for type parameters in MessageFrame 🤖 Generated with [Claude Code](https://claude.ai/code) * refactor: rename root to message
I've spent most of my day held up by this bug, see pydantic/pydantic-ai#1140 (comment) On python-sdk/src/mcp/shared/session.py Lines 344 to 346 in c897868
If I change these lines to if not isinstance(notification.root, LoggingMessageNotification):
await self._incoming_message_stream_writer.send(
notification
) Everything works correctly, obviously that's not a proper solution, but it demonstrates the problem. In case it helps, to reproduce this problem, run npx @pydantic/mcp-run-python sse (see pydantic/pydantic-ai#1140 for context) Then run import logging
from logging import basicConfig
from mcp import ClientSession, StdioServerParameters
from mcp.client.sse import sse_client
from mcp.client.stdio import stdio_client
from devtools import debug
import logfire
logfire.configure()
logfire.instrument_httpx(capture_all=True)
basicConfig(handlers=[logfire.LogfireLoggingHandler()], level=logging.DEBUG)
# disable httpx logs
logging.getLogger('httpcore.http11').setLevel(logging.CRITICAL)
logging.getLogger('httpx').setLevel(logging.CRITICAL)
logging.getLogger('urllib3.connectionpool').setLevel(logging.CRITICAL)
code = """
import numpy
a = numpy.array([1, 2, 3])
print(a)
a
"""
async def call_tools(session: ClientSession):
await session.initialize()
await session.set_logging_level('debug')
tools = await session.list_tools()
debug(tools)
result = await session.call_tool('run_python_code', {'python_code': code})
debug(result)
async def sse():
async with sse_client('http://localhost:3001/sse') as (read, write):
async with ClientSession(read, write) as session:
await call_tools(session)
async def stdio():
server_params = StdioServerParameters(command='npx', args=['@pydantic/mcp-run-python', 'stdio'])
async with stdio_client(server_params) as (read, write):
async with ClientSession(read, write) as session:
await call_tools(session)
if __name__ == '__main__':
import asyncio
with logfire.span('running sse'):
asyncio.run(sse()) Obviously you can disable the logfire bit, I was using it to see the raw http requests, I see the same hanging behaviour with both SSE and stdio. As I mentioned in the comment linked above, you can also comment out |
Those are the same three lines i identified, and I believe is deadlock. PR #202 is an end-to-end test that illustrates the condition. |
Fixes GitHub issue #201 by moving the incoming message stream and related methods from BaseSession to ServerSession where they are actually needed. This change follows the principle of having functionality only where it's required. GitHub-Issue:#201 🤖 Generated with [Claude Code](https://claude.ai/code)
Describe the bug
When a tool includes client-side logging with, the client seems to hang.
The following tool works ok with Claude Desktop and Inspector, but does not complete with a Python client.
To Reproduce
Will check in a PR with test and proposed fix.
Expected behavior
I would expect the tool to run to completion.
Screenshots
If applicable, add screenshots to help explain your problem.
Desktop (please complete the following information):
Smartphone (please complete the following information):
Additional context
Add any other context about the problem here.
The text was updated successfully, but these errors were encountered: