blob: ec5491e438f74a65c2e8f69e1c69fe31eb581726 [file] [log] [blame] [edit]
// Copyright 2025 The Go Authors. All rights reserved.
// Use of this source code is governed by a MIT
// license that can be found in the LICENSE file.
/*
* Project: grpc-go
* Issue or PR : https://github.com/grpc/grpc-go/pull/1275
* Buggy version: (missing)
* fix commit-id: 0669f3f89e0330e94bb13fa1ce8cc704aab50c9c
* Flaky: 100/100
*/
package main
import (
"io"
"os"
"runtime/pprof"
"time"
)
func init() {
register("Grpc1275", Grpc1275)
}
type recvBuffer_grpc1275 struct {
c chan bool
}
func (b *recvBuffer_grpc1275) get() <-chan bool {
return b.c
}
type recvBufferReader_grpc1275 struct {
recv *recvBuffer_grpc1275
}
func (r *recvBufferReader_grpc1275) Read(p []byte) (int, error) {
select {
case <-r.recv.get():
}
return 0, nil
}
type Stream_grpc1275 struct {
trReader io.Reader
}
func (s *Stream_grpc1275) Read(p []byte) (int, error) {
return io.ReadFull(s.trReader, p)
}
type http2Client_grpc1275 struct{}
func (t *http2Client_grpc1275) CloseStream(s *Stream_grpc1275) {
// It is the client.CloseSream() method called by the
// main goroutine that should send the message, but it
// is not. The patch is to send out this message.
}
func (t *http2Client_grpc1275) NewStream() *Stream_grpc1275 {
return &Stream_grpc1275{
trReader: &recvBufferReader_grpc1275{
recv: &recvBuffer_grpc1275{
c: make(chan bool),
},
},
}
}
func testInflightStreamClosing_grpc1275() {
client := &http2Client_grpc1275{}
stream := client.NewStream()
donec := make(chan bool)
go func() { // G2
defer close(donec)
stream.Read([]byte{1})
}()
client.CloseStream(stream)
timeout := time.NewTimer(300 * time.Nanosecond)
select {
case <-donec:
if !timeout.Stop() {
<-timeout.C
}
case <-timeout.C:
}
}
///
/// G1 G2
/// testInflightStreamClosing()
/// stream.Read()
/// io.ReadFull()
/// <- r.recv.get()
/// CloseStream()
/// <- donec
/// ------------G1 timeout, G2 leak---------------------
///
func Grpc1275() {
prof := pprof.Lookup("goroutineleak")
defer func() {
time.Sleep(100 * time.Millisecond)
prof.WriteTo(os.Stdout, 2)
}()
go func() {
testInflightStreamClosing_grpc1275() // G1
}()
}