-
Notifications
You must be signed in to change notification settings - Fork 38.4k
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
Intermittent Connection Reset with Spring HTTP Clients When Interacting with HTTP/1.0 Server #34228
Comments
Issue 1: Missing Content-Length HeaderYou'll need to wrap your request factory with Issue 2: Intermittent Connection ResetThere are a lot of moving parts here so it's hard to understand what you're seeing here. I don't think Spring is involved at the procotol level and none of questions you asked ring a bell. Could you share a sample application that reproduces the problem? Ideally, something we can git clone or that you can attach here as a zip. This should show the incorrect behavior with You can use https://hub.docker.com/r/hashicorp/http-echo as a container like so to get an HTTP1.0 server running:
Thanks! |
First, I created a basic HTTP 1.0 server implementation since the provided docker image didn't support HTTP 1.0 communication properly. Here's the code I used: @SpringBootApplication
class Application
fun main() {
val responseBody =
"""
{
"abcdefg1": 20,
"abcdefg2": "0101",
"abcdefg3": "0201",
"abcdefg4": "11ABCD1111",
"abcdefg5": 5,
"abcdefg6": true,
"abcdefg7": "20230422151239",
"abcdefg8": ""
}
""".trimIndent()
val server = ServerSocket(8080)
println("Server is running on port 8080...")
while (true) {
val client = server.accept()
try {
val reader = BufferedReader(InputStreamReader(client.inputStream))
while (true) {
val line = reader.readLine() ?: break
if (line.isEmpty()) break
}
val out = BufferedWriter(OutputStreamWriter(client.outputStream))
out.write("HTTP/1.0 200 OK\r\n")
out.write("Content-Type: application/json\r\n")
out.write("Connection: close\r\n")
out.write("\r\n")
out.write(responseBody)
out.flush()
} catch (e: Exception) {
println("Error: ${e.message}")
} finally {
try {
Thread.sleep(500)
client.close()
} catch (e: Exception) {
println("Error closing client: ${e.message}")
}
}
}
} Initially, I had mistakenly assumed that third-party libraries were handling this correctly. After recreating the issue with a basic HTTP 1.0 server implementation (with help from ChatGPT), I was able to reproduce the same errors I was experiencing. The key finding was understanding why third-party libraries like OkHttp3 and OpenFeign worked while Spring's HTTP clients didn't. It turns out that the success of other clients was illusory - they appeared to work because they included Content-Length headers, which prevented immediate connection closure. The root cause was in Spring's clients: The MappingJackson2HttpMessageConverter doesn't buffer the response before sending it, leading to issues when Content-Length is absent. The timing of socket closure seems to be directly related to the presence or absence of the Content-Length header. in the captured packets, the OK response arrived. Although it's not Spring's issue that the connection closed during serialization and couldn't be read, it's quite perplexing. |
As a side note, I don't think the process of wrapping with BufferingClientHttpRequestFactory was particularly clean. When using RestClient, rather than manually configuring beans, I utilized the powerful auto-configuration mechanism and received RestClient.Builder through Dependency Injection in the component, performing basic configuration in the constructor. However, since HttpClientProperties, ClientHttpRequestFactoryBuilder.detect(), and RestClientBuilderConfigurer involved in auto-configuration are loaded in global scope, I felt it was inconvenient that manual configuration was necessary in all cases where buffering was needed. Could my usual implementation style be problematic? I believe that Spring's auto-configuration settings are safer and more efficient than manually configuring based on internet searches. |
Now we've established that you couldn't really make this work with a client library but not with I think this discussion can be explained with the HTTP 1.0 RFC:
This means that a "Content-Length" header is required for all HTTP requests with a request body, when sent to an HTTP 1.0 server. This means that you must use the As for your latest comment about this request factory not being the default in Spring Boot auto-configuration: this is by design. Spring Framework was buffering by default in the past but we changed this in #30557 and called it in the migration guide:
While this is not ideal in your case, we believe that improving performance for the vast majority of application is worth a small inconvenience to applications still talking to HTTP 1.0 servers, which are quite rare these days. |
Spring HTTP Client Issues with HTTP/1.0 Server Integration
Issue Summary
Spring HTTP clients (WebClient, RestClient, RestTemplate) fail to properly receive data from the server, while the same requests work fine with HttpURLConnection, OpenFeign, and OkHttp3. Intermittent "Connection reset" exceptions occur during response reading.
Server Response Analysis (via Wireshark)
Steps to Reproduce
We're experiencing two distinct issues while integrating with a legacy HTTP/1.0 server that we cannot modify:
Issue 1: Missing Content-Length Header
Problem Statement
Current Behavior
Issue 2: Intermittent Connection Reset
Problem Statement
Current Behavior
Environment
Questions
Additional Information
The text was updated successfully, but these errors were encountered: