-
Notifications
You must be signed in to change notification settings - Fork 17.9k
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
net/http: ResponseController to manipulate per-request timeouts (and other behaviors) #54136
Comments
Change https://go.dev/cl/420174 mentions this issue: |
Thanks for the proposal. IIUC this will unblock removing timeoutHandler in kubernetes completely which has been a source of many hard to debug races in the past (kubernetes/kubernetes#105884). |
This proposal has been added to the active column of the proposals project |
If people want to understand a use case for this, at work we typically set a higher timeout duration for writes than reads (via http method.) The idea being that writes should have a higher likelihood of succeeding, and a faster timeout for reads forces engineers to work harder to make more efficient code. We haven’t implemented this in our go code because we couldn’t figure out how, the we do in other languages. |
Does anyone object to adding this API? |
@neild @rsc Just to clarify if I understand this correctly. This proposal would make the timeouts only possible per separate http.Handler, right? Routes in this case are not static and can be created, changed, deleted every ~3 seconds. The proxy serves a variety of applications next to each other. I hope my question is understandable. :) |
This proposal is for a mechanism to permit a handler to adjust the read and write deadlines for a request after the handler has been called. Sample usage would be something like:
|
@neild thanks! That was also my understanding, thanks for the clarification and example. |
Hello, also need this for streaming connections, a couple more questions to clarify:
|
Yes.
As proposed, this only works with My first thought was that the initial version of this proposal doesn't need to support third-party |
Changes to the proposal:
|
@neild I think this helps also my use case :) |
Change https://go.dev/cl/436890 mentions this issue: |
While working through a draft implementation of this, I came across a somewhat different approach to integrating non- In this approach,
A
We want to be able to add new methods to ResponseController in the future without breaking existing implementations. To ensure we can do this, the The advantage to this approach is less indirection: There's no need to pass each A disadvantage is that the pattern of embedding a default implementation of an interface with unexported methods is not commonly used in the standard library. However, this only affects people writing |
@neild, it sounds like there are two possible APIs here. Which one do you suggest? And does anyone else have any opinions on which one we use? |
After discussion with @bradfitz, we think the original proposal with the update in #54136 (comment) is the right approach. Making ResponseController an interface is an interesting idea, but the benefits (if any) are minor and the mechanisms used to make it work are non-obvious. |
Does anyone object to the API in #54136 (comment) ? |
For #41773 For #41773 For #50465 For #51914 For #53002 For #53896 For #53960 For #54136 For #54299 Change-Id: I729d5eafc1940d5706f980882a08ece1f69bb42c Reviewed-on: https://go-review.googlesource.com/c/go/+/450515 Auto-Submit: Damien Neil <[email protected]> TryBot-Result: Gopher Robot <[email protected]> Run-TryBot: Damien Neil <[email protected]> Reviewed-by: Ian Lance Taylor <[email protected]>
This will be in 1.20. |
Is it possible to somehow make deadlines be set also for individual At my $dayjob, we use long-lived HTTP connections (with chunked encoding) to transfer data ("soft-realtime"). These connections only end when either the client loses interest in receiving the data or the upstream shuts down, so a deadline for the full request has no sense—as basically there is no "end". To implement such policy, we have a custom type which wraps the So, can there be thought a way to implement such finer-grained controls? |
The You could also use |
Sorry, I forgot to state that we're currently using HTTP 1.1. |
the purpose of the above code is to extend the deadlines if we read/write a certain amount of bytes in a given time. This is required to protect long lived connections from Slowloris style attacks. I haven't tested the ResponseController yet but I think we can implement the same logic using it in combination with reader/writer wrappers. I'm not sure it's appropriate to simplify this use case directly into stdlib, but that would be great
|
Update net/http to enable tests that pass with the latest update to the vendored x/net. Update a few tests: Windows apparently doesn't guarantee that time.Since(time.Now()) is >=0, so to set a definitely-expired write deadline, use a time firmly in the past rather than now. Put a backoff loop on TestServerReadTimeout to avoid failures when the timeout expires mid-TLS-handshake. (The TLS handshake timeout is set to min(ReadTimeout, WriteTimeout, ReadHeaderTimeout); there's no way to set a long TLS handshake timeout and a short read timeout.) Don't close the http.Server in TestServerWriteTimeout while the handler may still be executing, since this can result in us getting the wrong error. Change the GOOS=js fake net implementation to properly return ErrDeadlineExceeded when a read/write deadline is exceeded, rather than EAGAIN. For #49837 For #54136 Change-Id: Id8a4ff6ac58336ff212dda3c8799b320cd6b9c19 Reviewed-on: https://go-review.googlesource.com/c/go/+/449935 Reviewed-by: Bryan Mills <[email protected]> Run-TryBot: Damien Neil <[email protected]> TryBot-Result: Gopher Robot <[email protected]>
This proposal won't affect the idle timeout between the requests ( Even if I use the
This is then overwritten by the
Do I get this right? Are there any workarounds / patterns to have a different per-connection idle timeout between the requests? |
Correct.
|
ResponseRecorder golang#54136 (implemented in Go 1.20) added the "http".ResponseController type, which allows manipulating per-request timeouts. This is especially useful for programs managing long-running HTTP connections such as Mercure. However, testing HTTP handlers leveraging per-request timeouts is currently cumbersome (even if doable) because "net/http/httptest".ResponseRecorder isn't compatible yet with "http".ResponseController. This patch makes ResponseRecorder compatible with "http".ResponseController. All new methods are part of the contract that response types must honor to be usable with "http".ResponseController. NewRecorderWithDeadlineAwareRequest() is necessary to test read deadlines, as calling rw.SetReadDeadline() must change the deadline on the request body. Fixes golang#60229. # Veuillez saisir le message de validation pour vos modifications. Les lignes # commençant par '#' seront conservées ; vous pouvez les supprimer vous-même # si vous le souhaitez. Un message vide abandonne la validation. # # Date : Mon May 15 17:59:56 2023 +0200 # # Sur la branche httptest_recorder_responsecontroller # Modifications qui seront validées : # modifié : net/http/httptest/example_test.go # modifié : net/http/httptest/recorder.go # modifié : net/http/httptest/recorder_test.go # modifié : net/http/server.go #
ResponseRecorder CL golang#54136 (implemented in Go 1.20) added the "http".ResponseController type, which allows manipulating per-request timeouts. This is especially useful for programs managing long-running HTTP connections such as Mercure. However, testing HTTP handlers leveraging per-request timeouts is currently cumbersome (even if doable) because "net/http/httptest".ResponseRecorder isn't compatible yet with "http".ResponseController. This patch makes ResponseRecorder compatible with "http".ResponseController. All new methods are part of the contract that response types must honor to be usable with "http".ResponseController. NewRecorderWithDeadlineAwareRequest() is necessary to test read deadlines, as calling rw.SetReadDeadline() must change the deadline on the request body. Fixes golang#60229.
…rder CL golang#54136 (implemented in Go 1.20) added the "http".ResponseController type, which allows manipulating per-request timeouts. This is especially useful for programs managing long-running HTTP connections such as Mercure. However, testing HTTP handlers leveraging per-request timeouts is currently cumbersome (even if doable) because "net/http/httptest".ResponseRecorder isn't compatible yet with "http".ResponseController. This patch makes ResponseRecorder compatible with "http".ResponseController. All new methods are part of the contract that response types must honor to be usable with "http".ResponseController. NewRecorderWithDeadlineAwareRequest() is necessary to test read deadlines, as calling rw.SetReadDeadline() must change the deadline on the request body. Fixes golang#60229.
Change https://go.dev/cl/495295 mentions this issue: |
This proposal seeks to address #16100 (no way of manipulating timeouts in Handler), and is inspired by #16100 (comment).
HTTP handler timeouts are specified on a per-Server basis:
ReadTimeout
,WriteTimeout
. It would be very useful to permit these timeouts (and possibly other options) to be overridden on a per-handler basis. For example, a handler which serves a long-running streaming response may want to extend theWriteTimeout
after each write, and will likely want a differentWriteTimeout
than a handler serving a short response.A problem is that we have no good place at the moment to add functions that adjust these timeouts. We might add methods to the
ResponseWriter
implementation and access them via type assertions (as is done with the existingFlush
andHijack
methods), but this proliferation of undiscoverable magic methods scales poorly and does not interact well with middleware which wraps theResponseWriter
type.Proposal: Add a new concrete
ResponseController
type which provides additional per-request controls. (AResponseWriter
writes the response, aResponseController
provides additional controls.) This type will supersede the existingFlusher
andHijacker
APIs.We additionally add the ability to set the read and write deadline on a per-request basis, via
ResponseController
. These functions take a deadline rather than a timeout, for consistency withnet.Conn
.The Handler returned by
http.TimeoutHandler
currently receives aResponseWriter
which does not implement theFlush
orHijack
methods. This will not change under this proposal: The*ResponseController
for aTimeoutHandler
will return a not-implemented error fromFlush
,Hijack
,SetWriteDeadline
, andSetReadDeadline
.The text was updated successfully, but these errors were encountered: