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

feat(client): Improve initialization state handling in McpAsyncClient #39

Merged
merged 6 commits into from
Mar 13, 2025

Conversation

tzolov
Copy link
Contributor

@tzolov tzolov commented Mar 11, 2025

  • Add proper initialization state tracking using AtomicBoolean and Sinks
  • Implement timeout handling for requests requiring initialization
  • Ensure all client methods verify initialization state before proceeding
  • Fix rootsListChangedNotification to check initialization state
  • Improve error messages for uninitialized client operations

- Add proper initialization state tracking using AtomicBoolean and Sinks
- Implement timeout handling for requests requiring initialization
- Ensure all client methods verify initialization state before proceeding
- Fix rootsListChangedNotification to check initialization state
- Improve error messages for uninitialized client operations

Signed-off-by: Christian Tzolov <[email protected]>
Copy link
Member

@chemicL chemicL left a comment

Choose a reason for hiding this comment

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

Left a couple suggestions, but overall this is a useful improvement :)

* @param operation The operation to execute if the client is initialized
* @return A Mono that completes with the result of the operation
*/
private <T> Mono<T> withInitializationCheck(String errorMessage,
Copy link
Member

Choose a reason for hiding this comment

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

"Client must be initialized before " is repeated in each use, consider replacing errorMessage with action and just specify what's about to happen, e.g. action="pinging the server".

Comment on lines 349 to 351
// --------------------------
// Utility Methods
// --------------------------
Copy link
Member

Choose a reason for hiding this comment

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

It's a not some "utility" but a fundamental prerequisite to almost any operation, perhaps we can group it under "initialization methods"?

Comment on lines 287 to 290
return this.mcpSession.sendNotification(McpSchema.METHOD_NOTIFICATION_INITIALIZED, null).doOnSuccess(v -> {
this.initialized.set(true);
this.initializedSink.tryEmitValue(initializeResult);
}).thenReturn(initializeResult);
Copy link
Member

Choose a reason for hiding this comment

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

Awesome, this will make the client more robust :)

@@ -149,6 +158,7 @@ public class McpAsyncClient {
this.clientCapabilities = features.clientCapabilities();
this.transport = transport;
this.roots = new ConcurrentHashMap<>(features.roots());
this.initializedTimeout = requestTimeout.multipliedBy(2);
Copy link
Member

Choose a reason for hiding this comment

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

Should it be this value and undocumented? I fear it would be worthwhile to add logging with a note after how long the client gave up. Also, this will have to be revisited once reconnects are considered.

@tzolov
Copy link
Contributor Author

tzolov commented Mar 11, 2025

Thanks for the feedback @chemicL . I've tried to address the review comments and to improve the documentation.

tzolov added 4 commits March 12, 2025 11:51
Signed-off-by: Christian Tzolov <[email protected]>
Signed-off-by: Christian Tzolov <[email protected]>
…ents

Replace hardcoded timeout constants with configurable getTimeoutDuration() method
Remove automatic initialization in setUp methods to allow explicit testing

Signed-off-by: Christian Tzolov <[email protected]>
Signed-off-by: Christian Tzolov <[email protected]>
@@ -92,8 +96,16 @@ void testConstructorWithInvalidArguments() {
.hasMessage("Request timeout must not be null");
}

@Test
void testListToolsWithoutInitialization() {
assertThatThrownBy(() -> mcpAsyncClient.listTools(null).block()).isInstanceOf(McpError.class)
Copy link
Member

@chemicL chemicL Mar 13, 2025

Choose a reason for hiding this comment

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

The test will last at least 600ms in case of WebClient and 4s for JDK HttpClient before the exception is observed, correct? If I am correct, it would be useful to consider using StepVerifier.withVirtualTime to emulate time passing by instead of adding more seconds to the time w validate the behaviour.

Copy link
Member

@chemicL chemicL 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 adding the tests. As a follow-up, it would be good to improve the feedback cycle and avoid blocking for several seconds to run unit tests as per my suggestion with virtual time verifier.

@tzolov tzolov merged commit 109aab2 into main Mar 13, 2025
@tzolov tzolov deleted the mcp-client-improve-init-sync branch March 13, 2025 14:13
tzolov added a commit that referenced this pull request Mar 17, 2025
…#39)

- Add proper initialization state tracking using AtomicBoolean and Sinks
- Implement timeout handling for requests requiring initialization
- Ensure all client methods verify initialization state before proceeding
- Fix rootsListChangedNotification to check initialization state
- Improve error messages for uninitialized client operations
- improve JavaDoc
- Add tests to verify proper error handling for uninitialized clients
  - Replace hardcoded timeout constants with configurable getTimeoutDuration() method
  - Remove automatic initialization in setUp methods to allow explicit testing

Signed-off-by: Christian Tzolov <[email protected]>
@tzolov tzolov added this to the 0.8.0 milestone Mar 21, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants