- 
                Notifications
    You must be signed in to change notification settings 
- Fork 810
feat(mcp): MCP response Span Capture for Stdio Mode #3383
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
          
     Closed
      
      
    
  
     Closed
                    Changes from 10 commits
      Commits
    
    
            Show all changes
          
          
            16 commits
          
        
        Select commit
          Hold shift + click to select a range
      
      dd87993
              
                mcpSpanIssueFix3364
              
              
                elinacse 42d8344
              
                codeClenup
              
              
                elinacse 97fe5df
              
                codeClenup
              
              
                elinacse 0f76f4f
              
                codeCleanup
              
              
                elinacse 85ae1a2
              
                Merge branch 'main' into mcpEnableObservability
              
              
                elinacse ee24f3d
              
                mcpSpanIssueFix3364
              
              
                elinacse 93b8675
              
                codeClenup
              
              
                elinacse b61d00f
              
                mcpStdioFix
              
              
                elinacse 4909c97
              
                testcaseAdded
              
              
                elinacse 06a9498
              
                utFix
              
              
                elinacse afe2da6
              
                addErrorReason
              
              
                elinacse 0ad5895
              
                resolvedConflicts
              
              
                elinacse e4f54ca
              
                codeCleanup
              
              
                elinacse 035dd4a
              
                mcpChanges
              
              
                elinacse f4eb666
              
                codeCleanup
              
              
                elinacse 7b21c59
              
                Merge branch 'main' into mcpEnableObservability
              
              
                elinacse File filter
Filter by extension
Conversations
          Failed to load comments.   
        
        
          
      Loading
        
  Jump to
        
          Jump to file
        
      
      
          Failed to load files.   
        
        
          
      Loading
        
  Diff view
Diff view
          Some comments aren't visible on the classic Files Changed page.
        
There are no files selected for viewing
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
              
        
          
          
            114 changes: 114 additions & 0 deletions
          
          114 
        
  packages/opentelemetry-instrumentation-mcp/tests/test_jsonrpc_response_wrapper.py
  
  
      
      
   
        
      
      
    
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
              | Original file line number | Diff line number | Diff line change | 
|---|---|---|
| @@ -0,0 +1,114 @@ | ||
| """Test cases for _jsonrpc_response_init_wrapper method demonstrating stdio MCP use cases. | ||
|  | ||
| These tests demonstrate how the JSONRPC response wrapper works with MCP communication, | ||
| which is the same pattern used by external clients like Claude and Copilot when they | ||
| communicate with MCP servers via stdio (standard input/output). | ||
|  | ||
| Note: Due to stdio communication complexity in test environments, these tests use | ||
| FastMCP which implements the same JSONRPC patterns that stdio mode uses. | ||
| """ | ||
|  | ||
| from opentelemetry.semconv_ai import SpanAttributes | ||
| from opentelemetry.trace.status import StatusCode | ||
|  | ||
|  | ||
| async def test_jsonrpc_response_stdio_success(span_exporter, tracer_provider) -> None: | ||
| """Test JSONRPC response wrapper with successful MCP tool execution. | ||
|  | ||
| This demonstrates the same JSONRPC communication pattern that external clients | ||
| like Claude and Copilot use when communicating with MCP servers via stdio. | ||
| The _jsonrpc_response_init_wrapper captures these responses and creates spans. | ||
| """ | ||
| from fastmcp import FastMCP, Client | ||
|  | ||
| # Create a FastMCP server (implements JSONRPC patterns as stdio mode) | ||
| server = FastMCP("test-server") | ||
|  | ||
| @server.tool() | ||
| async def test_tool(message: str) -> str: | ||
| """A test tool that returns a message. | ||
|  | ||
| In stdio mode, this would be called by external clients | ||
| like Claude/Copilot via JSONRPC over stdin/stdout. | ||
| """ | ||
| return f"Tool executed successfully: {message}" | ||
|  | ||
| # Use in-memory client (same JSONRPC patterns as stdio_client) | ||
| async with Client(server) as client: | ||
| # This simulates the JSONRPC request/response cycle that happens | ||
| # when external clients call MCP tools via stdio | ||
| result = await client.call_tool("test_tool", {"message": "Hello from stdio client"}) | ||
| assert len(result) == 1 | ||
| assert "Tool executed successfully: Hello from stdio client" in result[0].text | ||
|  | ||
| # Get the finished spans | ||
| spans = span_exporter.get_finished_spans() | ||
|  | ||
| # Look for MCP_Tool_Response spans created by the JSONRPC response wrapper | ||
| mcp_response_spans = [span for span in spans if span.name == "MCP_Tool_Response"] | ||
|  | ||
| assert len(mcp_response_spans) >= 1, f"Expected at least 1 MCP_Tool_Response span, found {len(mcp_response_spans)}" | ||
|  | ||
| response_span = mcp_response_spans[0] | ||
| assert response_span.status.status_code == StatusCode.OK | ||
|  | ||
| assert SpanAttributes.MCP_RESPONSE_VALUE in response_span.attributes | ||
| response_value = response_span.attributes[SpanAttributes.MCP_RESPONSE_VALUE] | ||
| assert "Tool executed successfully" in response_value | ||
|  | ||
| assert SpanAttributes.MCP_REQUEST_ID in response_span.attributes | ||
| request_id = response_span.attributes[SpanAttributes.MCP_REQUEST_ID] | ||
| assert request_id is not None | ||
|  | ||
|  | ||
| async def test_jsonrpc_response_stdio_error(span_exporter, tracer_provider) -> None: | ||
| """Test JSONRPC response wrapper with MCP error response. | ||
|  | ||
| This demonstrates error handling in the same JSONRPC communication pattern | ||
| that external clients like Claude and Copilot use when communicating with | ||
| MCP servers via stdio. | ||
| """ | ||
| from fastmcp import FastMCP, Client | ||
|  | ||
| # Create a FastMCP server (implements same JSONRPC patterns as stdio mode) | ||
| server = FastMCP("test-server") | ||
|  | ||
| @server.tool() | ||
| async def failing_tool(should_fail: bool = True) -> str: | ||
| """A tool that can fail for testing error handling. | ||
|  | ||
| In stdio mode, this error would be captured and sent back | ||
| to external clients like Claude/Copilot via JSONRPC error response. | ||
| """ | ||
| if should_fail: | ||
| raise ValueError("Intentional test error - simulating stdio error response") | ||
| return "Success!" | ||
|  | ||
| # Use in-memory client (same JSONRPC patterns as stdio_client) | ||
| async with Client(server) as client: | ||
| # This simulates the JSONRPC error response cycle that happens | ||
| # when external clients call MCP tools via stdio and they fail | ||
| try: | ||
| await client.call_tool("failing_tool", {"should_fail": True}) | ||
| except Exception: | ||
| pass # Expected to fail - this simulates stdio error handling | ||
|  | ||
| # Get the finished spans | ||
| spans = span_exporter.get_finished_spans() | ||
|  | ||
| # Look for MCP_Tool_Response spans created by the JSONRPC response wrapper | ||
| mcp_response_spans = [span for span in spans if span.name == "MCP_Tool_Response"] | ||
|  | ||
| # Should have at least one MCP_Tool_Response span | ||
| assert len(mcp_response_spans) >= 1, f"Expected at least 1 MCP_Tool_Response span, found {len(mcp_response_spans)}" | ||
|  | ||
| # Check the first MCP_Tool_Response span | ||
| response_span = mcp_response_spans[0] | ||
| assert response_span.status.status_code == StatusCode.ERROR | ||
| assert response_span.status.description == "Tool execution error" | ||
|  | ||
| assert SpanAttributes.MCP_RESPONSE_VALUE in response_span.attributes | ||
| response_value = response_span.attributes[SpanAttributes.MCP_RESPONSE_VALUE] | ||
| assert "error" in response_value.lower() or "Intentional test error" in response_value | ||
|  | ||
| assert SpanAttributes.MCP_REQUEST_ID in response_span.attributes | 
  Add this suggestion to a batch that can be applied as a single commit.
  This suggestion is invalid because no changes were made to the code.
  Suggestions cannot be applied while the pull request is closed.
  Suggestions cannot be applied while viewing a subset of changes.
  Only one suggestion per line can be applied in a batch.
  Add this suggestion to a batch that can be applied as a single commit.
  Applying suggestions on deleted lines is not supported.
  You must change the existing code in this line in order to create a valid suggestion.
  Outdated suggestions cannot be applied.
  This suggestion has been applied or marked resolved.
  Suggestions cannot be applied from pending reviews.
  Suggestions cannot be applied on multi-line comments.
  Suggestions cannot be applied while the pull request is queued to merge.
  Suggestion cannot be applied right now. Please check back later.
  
    
  
    
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In _jsonrpc_response_init_wrapper, wrapping the serialized result in an f-string is redundant, and using args[0] as a fallback for id_value assumes a specific init signature. Please document or validate the expected parameter order.