For stdio-based MCP servers, stdin closing is effectively the client disconnect signal. This is especially important when an MCP server runs in a container: once the MCP client disconnects or closes stdin, the server should be able to trigger application shutdown and clean up background resources.
Today, StdioServerTransportProvider closes the MCP session when stdin reaches EOF, but it does not expose a public transport-level callback that applications can use to run shutdown logic.
Downstream, we had to vendor/copy StdioServerTransportProvider to add such a callback.
Current behavior
When the stdin read loop exits, the SDK transport does roughly this:
finally {
isClosing.set(true);
if (session != null) {
session.close();
}
inboundSink.tryEmitComplete();
}
The session is closed, but the application has no direct hook to react to the stdio client disconnect.
There is also a subtle ordering issue. handleIncomingMessages() currently disposes the inbound scheduler from doOnTerminate():
this.inboundSink.asFlux()
.flatMap(message -> session.handle(message))
.doOnTerminate(() -> {
this.outboundSink.tryEmitComplete();
this.inboundScheduler.dispose();
})
.subscribe();
When stdin closes, the inbound read loop completes inboundSink. The termination callback can run on the same inbound thread. Disposing the scheduler there may call shutdownNow(), interrupting that same thread before downstream shutdown work has completed.
We observed this downstream: shutdown logic could run with the current thread interrupt flag already set, causing graceful shutdown to fail or exit early.
Expected behavior
StdioServerTransportProvider should allow applications to register an optional callback for stdin EOF / stdio client disconnect.
That callback should run:
- After the session is closed.
- After the inbound sink is completed.
- Before the inbound scheduler is disposed.
- Without the current thread being interrupted by scheduler disposal.
Proposed fix
Add an optional callback to StdioServerTransportProvider, for example as a Runnable:
public StdioServerTransportProvider(
McpJsonMapper jsonMapper,
InputStream inputStream,
OutputStream outputStream,
Runnable closeCallback
)
Or, if a reactive API is preferred:
Supplier<Mono<Void>> closeCallback
Then move inboundScheduler.dispose() out of handleIncomingMessages().doOnTerminate(...) and into the inbound read-loop finally, after the callback has completed.
The inbound read-loop cleanup would look conceptually like this:
finally {
isClosing.set(true);
if (session != null) {
session.close();
}
inboundSink.tryEmitComplete();
if (closeCallback != null) {
closeCallback.run();
}
inboundScheduler.dispose();
}
handleIncomingMessages() should still complete the outbound sink, but should not dispose the inbound scheduler from the termination callback.
Suggested regression test
Add a test that:
- Creates a
StdioServerTransportProvider with a piped input stream.
- Registers a callback.
- Starts the transport with a mock session.
- Closes the piped output stream to simulate stdin EOF.
- Verifies the callback runs.
- Verifies
Thread.currentThread().isInterrupted() is false inside the callback.
For stdio-based MCP servers,
stdinclosing is effectively the client disconnect signal. This is especially important when an MCP server runs in a container: once the MCP client disconnects or closes stdin, the server should be able to trigger application shutdown and clean up background resources.Today,
StdioServerTransportProvidercloses the MCP session when stdin reaches EOF, but it does not expose a public transport-level callback that applications can use to run shutdown logic.Downstream, we had to vendor/copy
StdioServerTransportProviderto add such a callback.Current behavior
When the stdin read loop exits, the SDK transport does roughly this:
The session is closed, but the application has no direct hook to react to the stdio client disconnect.
There is also a subtle ordering issue.
handleIncomingMessages()currently disposes the inbound scheduler fromdoOnTerminate():When stdin closes, the inbound read loop completes
inboundSink. The termination callback can run on the same inbound thread. Disposing the scheduler there may callshutdownNow(), interrupting that same thread before downstream shutdown work has completed.We observed this downstream: shutdown logic could run with the current thread interrupt flag already set, causing graceful shutdown to fail or exit early.
Expected behavior
StdioServerTransportProvidershould allow applications to register an optional callback for stdin EOF / stdio client disconnect.That callback should run:
Proposed fix
Add an optional callback to
StdioServerTransportProvider, for example as aRunnable:Or, if a reactive API is preferred:
Then move
inboundScheduler.dispose()out ofhandleIncomingMessages().doOnTerminate(...)and into the inbound read-loopfinally, after the callback has completed.The inbound read-loop cleanup would look conceptually like this:
handleIncomingMessages()should still complete the outbound sink, but should not dispose the inbound scheduler from the termination callback.Suggested regression test
Add a test that:
StdioServerTransportProviderwith a piped input stream.Thread.currentThread().isInterrupted()isfalseinside the callback.