diff --git a/shared_client.go b/shared_client.go index 621402fcf..a4375f708 100644 --- a/shared_client.go +++ b/shared_client.go @@ -304,7 +304,9 @@ func (c *SharedClient) ExchangeSharedContext(ctx context.Context, m *Msg) (r *Ms timeout := c.getTimeoutForRequest(c.Client.writeTimeout()) ctx, cancel := context.WithTimeout(ctx, timeout) defer cancel() - respCh := make(chan sharedClientResponse) + // The handler loop writes our response or an error to this channel. If we fall into the timeout + // below, the handling loop would block indefinitely on writing if this channel is not buffered. + respCh := make(chan sharedClientResponse, 1) select { case c.requests <- request{ctx: ctx, msg: m, ch: respCh}: case <-ctx.Done(): @@ -315,7 +317,8 @@ func (c *SharedClient) ExchangeSharedContext(ctx context.Context, m *Msg) (r *Ms select { case resp := <-respCh: return resp.msg, resp.rtt, resp.err - // This is just fail-safe mechanism in case there is another similar issue + // This is a fail-safe mechanism which prevents leaking this goroutine if the handling loop + // fails to write a response on our channel. case <-time.After(time.Minute): return nil, 0, fmt.Errorf("timeout waiting for response") }