- 
                Notifications
    You must be signed in to change notification settings 
- Fork 546
          Restructure IMcpEndpoint, IMcpClient, and IMcpServer
          #723
        
          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
  
    Restructure IMcpEndpoint, IMcpClient, and IMcpServer
  
  #723
              Conversation
| One downside to this renaming is that it distances the SDK from the TypeScript and Python SDKs, which makes it harder for developer to transition directly between. It's a very minor drift - adding postfixing -Session to McpClient. But I still think it's worth considering every time drift is introduced whether it is strictly necessary or a unified naming can be kept. It is also preferable, I think, if McpClient aligns directly with the Client concept as described in: https://modelcontextprotocol.io/specification/2025-06-18/architecture. I was never happy with McpClientFactory so I am glad to see it go, but definitely don't think it should be renamed to McpClient, as that was cause conceptual confusion with the Client concept in the specification. A client is a client session within a host. Restructuring is definitely warranted, but I'd like to caution against drift from the other SDKs. There's a lot of polyglot development going in this space, and there are many upsides to having at least the specification concepts named identically in the SDKs. With this change the SDK uses McpClient to mean something other than the specification and the other SDKs and uses McpClientSession to refer to the concept of a client. I can see the point of using the latter, but the former I am skeptical of. A client is a very well defined concept in the protocol spec and in the other SDKs. | 
IMcpEndpoint, IMcpClient, and IMcpServerIMcpEndpoint, IMcpClient, and IMcpServer
      | Really like the revised naming. It removes the Factory and Endpoint abstractions, and does so without drifting from the concepts used in the other SDKs and the specification docs. This version is easier to understand than the existing code for someone coming from another SDK or from the docs - in addition to accomplishing the needed restructuring. | 
| Really appreicate the feedback, @PederHP! I agree that the previous naming could have created confusion since, as you mentioned, the distinction between a "client" and a "client session" doesn't really exist in the MCP spec. And developers might be surprised if "McpClient" means something different in the C# SDK than SDKs for other langs. | 
| We should also consider renaming  My proposal would be to call it  | 
| 
 I agree that it should be renamed. I was drafting a PR at one point to align with the other SDKs but then got sidetracked and kinda forgot about it. There's a discrepancy where the C# SDK uses SseClientTransport as the entry point for both SSE and Streaming HTTP. I've had to help multiple developers, online and my day job, on how to handle config-based clients, because the other SDKs have three different Transport types and just map  I know that using the transport types directly is much less relevant after this change, but I still think renaming the transport and related types would be helpful in making the SDK easier to use for polyglot developers. | 
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.
I think all the renames are clear improvements over the older names. I still think having "ConnectAsync" and "LaunchAsync" methods that don't require you to initialize a transport to get a client could be nice, but that could be a follow up.
I'm curious on how long we plan to keep around the obsoleted types and methods. I hope we can fully delete them by 1.0, but I see the value in keeping them for a few preview releases at least to make it easier to upgrade.
I'd also prefer to keep something like the McpEndpint._disposeLock SemaphoreSlim for similar reasons we have KestrelServerImple._stoppedTcs. It's not exactly the same because Kestrel waits for middleware to complete during shutdown while the _messageProcessingTask does not currently wait on any handlers, but I don't think it's much more complicated than the CompareExchange on _isDisposed, and it's arguably more correct especially if we do ever end up waiting on handlers.
        
          
                src/ModelContextProtocol.Core/Server/AugmentedServiceProvider.cs
              
                Outdated
          
            Show resolved
            Hide resolved
        
      Co-authored-by: Stephen Halter <[email protected]>
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.
Thanks, good change overall.
| Breaking Changes (Migration Summary)Update your code for the following renamed and restructured APIs. 1. Interfaces replaced by abstract classesReplace usages of the obsolete interfaces with the new abstract types: 
 Action: Change variable, field, and parameter types. (Optional but recommended: rename identifiers like  2. Factory classes deprecated
 Action: Replace calls to the factory classes with the static factory methods on the abstract classes. 3. Transport rename
 Action: Update usage of these types to utilize the new type names. | 
Breaking Changes (Migration Summary)
Update your code for the following renamed and restructured APIs.
1. Interfaces replaced by abstract classes
Replace usages of the obsolete interfaces with the new abstract types:
IMcpClientMcpClientIMcpServerMcpServerIMcpEndpointMcpSessionAction: Change variable, field, and parameter types. (Optional but recommended: rename identifiers like
endpoint→sessionfor clarity.) The obsolete interfaces remain temporarily with[Obsolete]warnings.2. Factory classes deprecated
McpClientFactory.CreateAsync(...)McpClient.CreateAsync(...)McpServerFactory.Create(...)McpServer.Create(...)Action: Replace calls to the factory classes with the static factory methods on the abstract classes.
3. Transport rename
SseClientTransportHttpClientTransportSseClientTransportOptionsHttpClientTransportOptionsAction: Update usage of these types to utilize the new type names.
Additional Changes
McpClient->McpClientImplMcpServer->McpServerImplMcpSession->McpSessionHandlerMcpEndpointabstract classIMcpClient,IMcpServer, andIMcpEndpointand places them directly inside their respective new abstract classesMcpClientImplandMcpServerImplto function without relying on a common base class implementing shared functionalityNote
See the edit history of this comment to view previous revisions
Description
This change generally aligns with the suggestions in this comment.
An alternative could be to remove
McpClientfactory methods altogether and do something similar to what was suggested here. However, this means that we're differentiating between a "client" and a "client session", and the MCP specification doesn't make such a distinction. In the future, we could still add new factory methods that makeMcpClientcreation more concise. For example:Or:
Fixes #355