Skip to content

Commit d55ebd8

Browse files
authored
Make stream client closers non-blocking (#791)
This updates the behavior of the streaming client methods `BidiStreamForClient.CloseResponse` and `ServerStreamForClient.Close` to be non-blocking, aligning it with the standard behavior of `net/http`'s `Request.Body` closure. Previously, the implementation used a graceful, blocking closure that fully read from the stream before closing. This allows for reuse of the underlying TCP connection. However, this behavior could lead to unexpected client hangs, as users may not anticipate blocking on close. To address this, the closers no longer drain the stream. Documentation has been updated to clarify the behavior and provide users a workaround to keep the optimization by calling receive until the stream is drained. This avoids unexpected blocking behavior in client applications. Fixes #789 Signed-off-by: Edward McFarlane <[email protected]>
1 parent 7dc3e6d commit d55ebd8

File tree

2 files changed

+10
-7
lines changed

2 files changed

+10
-7
lines changed

client_stream.go

+9
Original file line numberDiff line numberDiff line change
@@ -163,6 +163,10 @@ func (s *ServerStreamForClient[Res]) ResponseTrailer() http.Header {
163163
}
164164

165165
// Close the receive side of the stream.
166+
//
167+
// Close is non-blocking. To gracefully close the stream and allow for
168+
// connection resuse ensure all messages have been received before calling
169+
// Close. All messages are received when Receive returns false.
166170
func (s *ServerStreamForClient[Res]) Close() error {
167171
if s.constructErr != nil {
168172
return s.constructErr
@@ -251,6 +255,11 @@ func (b *BidiStreamForClient[Req, Res]) Receive() (*Res, error) {
251255
}
252256

253257
// CloseResponse closes the receive side of the stream.
258+
//
259+
// CloseResponse is non-blocking. To gracefully close the stream and allow for
260+
// connection resuse ensure all messages have been received before calling
261+
// CloseResponse. All messages are received when Receive returns an error
262+
// wrapping [io.EOF].
254263
func (b *BidiStreamForClient[Req, Res]) CloseResponse() error {
255264
if b.err != nil {
256265
return b.err

duplex_http_call.go

+1-7
Original file line numberDiff line numberDiff line change
@@ -239,13 +239,7 @@ func (d *duplexHTTPCall) CloseRead() error {
239239
if d.response == nil {
240240
return nil
241241
}
242-
_, err := discard(d.response.Body)
243-
closeErr := d.response.Body.Close()
244-
if err == nil ||
245-
errors.Is(err, context.Canceled) ||
246-
errors.Is(err, context.DeadlineExceeded) {
247-
err = closeErr
248-
}
242+
err := d.response.Body.Close()
249243
err = wrapIfContextDone(d.ctx, err)
250244
return wrapIfRSTError(err)
251245
}

0 commit comments

Comments
 (0)