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: enable HTTP redirect handling in SSE client #284

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

eastlondoner
Copy link

@eastlondoner eastlondoner commented Mar 14, 2025

Updated the httpx AsyncClient to explicitly set follow_redirects=True to handle redirect responses from remote MCP servers. This fixes an issue where connections to servers that return redirects (like yo-mcp.com) would fail.

Motivation and Context

When connecting to certain MCP servers (like yo-mcp.com) that respond with HTTP redirects, the client would fail to establish a connection. This is because the redirect wasn't being followed automatically by the SSE client. The TypeScript client handles this correctly, but the Python client needed an explicit configuration.

How Has This Been Tested?

  • Added a test script (scripts/test_redirect.py) that connects to https://yo-mcp.com/mcp/7NIUz0cvfStY6-wSsVyY3 and verifies the redirect handling works correctly
  • Ran the script which confirmed successful connection with the server, including following the redirect and completing session initialization
  • Verified in logs that the client correctly follows the HTTP 302 redirect

Breaking Changes

None. This is a non-breaking bug fix that enables better compatibility with remote servers.

Types of changes

  • Bug fix (non-breaking change which fixes an issue)
  • New feature (non-breaking change which adds functionality)
  • Breaking change (fix or feature that would cause existing functionality to change)
  • Documentation update

Checklist

  • I have read the MCP Documentation
  • My code follows the repository's style guidelines
  • New and existing tests pass locally
  • I have added appropriate error handling
  • I have added or updated documentation as needed

Additional context

The issue was simple - httpx has follow_redirects=True by default, but for some reason the redirect wasn't being followed by the SSE client. Adding the explicit parameter resolves the issue. The test script shows how to connect to a remote MCP server and can be used as a reference for other developers.

Updated the httpx AsyncClient to explicitly set follow_redirects=True to handle
redirect responses from remote MCP servers. This fixes an issue where connections
to servers that return redirects (like yo-mcp.com) would fail.

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <[email protected]>
Copy link

@gruckion gruckion left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good, is there going to be another option needed? Shouldn’t we enable a way to provide a set of options to inject in?

@@ -43,7 +43,7 @@ async def sse_client(
async with anyio.create_task_group() as tg:
try:
logger.info(f"Connecting to SSE endpoint: {remove_request_params(url)}")
async with httpx.AsyncClient(headers=headers) as client:
async with httpx.AsyncClient(headers=headers, follow_redirects=True) as client:

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We have multiple spots where we create an AsyncClient. We should introduce a function to create a stnadard AsyncClient with respective options.

@eastlondoner
Copy link
Author

I can’t really envision a scenario where you wouldn’t want to follow redirects since it’s a core part of the http protocol. I think adding options is YAGNI but if you think it’s necessary I can prompt claude to fix it.

many pointers on what I should do to get this PR green?

@allenporter
Copy link
Contributor

There are lint errors that should be fixed.

My take is I don't think this PR should add more options.

Copy link
Member

@dsp-ant dsp-ant left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for looking into this. Happy to accept once the lint errors are fixed.

Note that I think it would be preferable if we had a function somewhere that creates an AsynClient object with whatever options we choose to be the default for the MCP SDK.

@@ -43,7 +43,7 @@ async def sse_client(
async with anyio.create_task_group() as tg:
try:
logger.info(f"Connecting to SSE endpoint: {remove_request_params(url)}")
async with httpx.AsyncClient(headers=headers) as client:
async with httpx.AsyncClient(headers=headers, follow_redirects=True) as client:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We have multiple spots where we create an AsyncClient. We should introduce a function to create a stnadard AsyncClient with respective options.

@Kludex
Copy link
Member

Kludex commented Mar 17, 2025

I think we can pass the client, instead of using a factory.

@dsp-ant
Copy link
Member

dsp-ant commented Mar 17, 2025

I think we can pass the client, instead of using a factory.

that works too, but we need to make sure we have a central creation point to not have to sync options across client object creations.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants