Skip to content

Commit 729e688

Browse files
Make client config optional to McpClientFactory.CreateAsync (#65)
* Make client config optional to McpClientFactory.CreateAsync Unless you're specifying capabilities (sampling/rooting) or need a specific client name/version, the client information can be inferred from the assembly, simplifying the getting-started experience. * Cache default options * Update src/ModelContextProtocol/Client/McpClientFactory.cs Co-authored-by: Eirik Tsarpalis <[email protected]> --------- Co-authored-by: Eirik Tsarpalis <[email protected]>
1 parent 36d5019 commit 729e688

File tree

4 files changed

+50
-23
lines changed

4 files changed

+50
-23
lines changed

README.MD

+3-11
Original file line numberDiff line numberDiff line change
@@ -20,16 +20,10 @@ For more information about MCP:
2020
## Getting Started (Client)
2121

2222
To get started writing a client, the `McpClientFactory.CreateAsync` method is used to instantiate and connect an `IMcpClient`
23-
to a server, with details about the client and server specified in `McpClientOptions` and `McpServerConfig` objects.
24-
Once you have an `IMcpClient`, you can interact with it, such as to enumerate all available tools and invoke tools.
23+
to a server. Once you have an `IMcpClient`, you can interact with it, such as to enumerate all available tools and invoke tools.
2524

2625
```csharp
27-
McpClientOptions options = new()
28-
{
29-
ClientInfo = new() { Name = "TestClient", Version = "1.0.0" }
30-
};
31-
32-
McpServerConfig config = new()
26+
var client = await McpClientFactory.CreateAsync(new()
3327
{
3428
Id = "everything",
3529
Name = "Everything",
@@ -39,9 +33,7 @@ McpServerConfig config = new()
3933
["command"] = "npx",
4034
["arguments"] = "-y @modelcontextprotocol/server-everything",
4135
}
42-
};
43-
44-
var client = await McpClientFactory.CreateAsync(config, options);
36+
});
4537

4638
// Print the list of tools available from the server.
4739
await foreach (var tool in client.ListToolsAsync())

samples/ChatWithTools/Program.cs

+1-2
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,7 @@
1515
{
1616
["command"] = "npx", ["arguments"] = "-y @modelcontextprotocol/server-everything",
1717
}
18-
},
19-
new() { ClientInfo = new() { Name = "ChatClient", Version = "1.0.0" } });
18+
});
2019

2120
// Get all available tools
2221
Console.WriteLine("Tools available:");

src/ModelContextProtocol/Client/McpClientFactory.cs

+24-3
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,36 @@
66
using ModelContextProtocol.Utils;
77
using Microsoft.Extensions.Logging;
88
using Microsoft.Extensions.Logging.Abstractions;
9+
using System.Reflection;
910

1011
namespace ModelContextProtocol.Client;
1112

1213
/// <summary>Provides factory methods for creating MCP clients.</summary>
1314
public static class McpClientFactory
1415
{
16+
/// <summary>Default client options to use when none are supplied.</summary>
17+
private static readonly McpClientOptions s_defaultClientOptions = CreateDefaultClientOptions();
18+
19+
/// <summary>Creates default client options to use when no options are supplied.</summary>
20+
private static McpClientOptions CreateDefaultClientOptions()
21+
{
22+
var asmName = (Assembly.GetEntryAssembly() ?? Assembly.GetExecutingAssembly()).GetName();
23+
return new()
24+
{
25+
ClientInfo = new()
26+
{
27+
Name = asmName.Name ?? "McpClient",
28+
Version = asmName.Version?.ToString() ?? "1.0.0",
29+
},
30+
};
31+
}
32+
1533
/// <summary>Creates an <see cref="IMcpClient"/>, connecting it to the specified server.</summary>
1634
/// <param name="serverConfig">Configuration for the target server to which the client should connect.</param>
17-
/// <param name="clientOptions">A client configuration object which specifies client capabilities and protocol version.</param>
35+
/// <param name="clientOptions">
36+
/// A client configuration object which specifies client capabilities and protocol version.
37+
/// If <see langword="null"/>, details based on the current process will be employed.
38+
/// </param>
1839
/// <param name="createTransportFunc">An optional factory method which returns transport implementations based on a server configuration.</param>
1940
/// <param name="loggerFactory">A logger factory for creating loggers for clients.</param>
2041
/// <param name="cancellationToken">A token to cancel the operation.</param>
@@ -25,14 +46,14 @@ public static class McpClientFactory
2546
/// <exception cref="InvalidOperationException"><paramref name="createTransportFunc"/> returns an invalid transport.</exception>
2647
public static async Task<IMcpClient> CreateAsync(
2748
McpServerConfig serverConfig,
28-
McpClientOptions clientOptions,
49+
McpClientOptions? clientOptions = null,
2950
Func<McpServerConfig, ILoggerFactory?, IClientTransport>? createTransportFunc = null,
3051
ILoggerFactory? loggerFactory = null,
3152
CancellationToken cancellationToken = default)
3253
{
3354
Throw.IfNull(serverConfig);
34-
Throw.IfNull(clientOptions);
3555

56+
clientOptions ??= s_defaultClientOptions;
3657
createTransportFunc ??= CreateTransport;
3758

3859
string endpointName = $"Client ({serverConfig.Id}: {serverConfig.Name})";

tests/ModelContextProtocol.Tests/Client/McpClientFactoryTests.cs

+22-7
Original file line numberDiff line numberDiff line change
@@ -19,13 +19,6 @@ public async Task CreateAsync_WithInvalidArgs_Throws()
1919
{
2020
await Assert.ThrowsAsync<ArgumentNullException>("serverConfig", () => McpClientFactory.CreateAsync((McpServerConfig)null!, _defaultOptions, cancellationToken: TestContext.Current.CancellationToken));
2121

22-
await Assert.ThrowsAsync<ArgumentNullException>("clientOptions", () => McpClientFactory.CreateAsync(new McpServerConfig()
23-
{
24-
Name = "name",
25-
Id = "id",
26-
TransportType = TransportTypes.StdIo,
27-
}, (McpClientOptions)null!, cancellationToken: TestContext.Current.CancellationToken));
28-
2922
await Assert.ThrowsAsync<ArgumentException>("serverConfig", () => McpClientFactory.CreateAsync(new McpServerConfig()
3023
{
3124
Name = "name",
@@ -41,6 +34,28 @@ await Assert.ThrowsAsync<InvalidOperationException>(() => McpClientFactory.Creat
4134
}, _defaultOptions, (_, __) => null!, cancellationToken: TestContext.Current.CancellationToken));
4235
}
4336

37+
[Fact]
38+
public async Task CreateAsync_NullOptions_EntryAssemblyInferred()
39+
{
40+
// Arrange
41+
var serverConfig = new McpServerConfig
42+
{
43+
Id = "test-server",
44+
Name = "Test Server",
45+
TransportType = TransportTypes.StdIo,
46+
Location = "/path/to/server",
47+
};
48+
49+
// Act
50+
await using var client = await McpClientFactory.CreateAsync(
51+
serverConfig,
52+
null,
53+
(_, __) => new NopTransport(),
54+
cancellationToken: TestContext.Current.CancellationToken);
55+
56+
Assert.NotNull(client);
57+
}
58+
4459
[Fact]
4560
public async Task CreateAsync_WithValidStdioConfig_CreatesNewClient()
4661
{

0 commit comments

Comments
 (0)