fix memleak in safe.Pool

Co-authored-by: Julien Salleyron <julien.salleyron@gmail.com>
This commit is contained in:
mpl 2020-01-20 17:42:05 +01:00 committed by Traefiker Bot
parent f84d947115
commit 24192a3797
3 changed files with 8 additions and 109 deletions

View file

@ -19,14 +19,13 @@ type routineCtx func(ctx context.Context)
// Pool is a pool of go routines // Pool is a pool of go routines
type Pool struct { type Pool struct {
routines []routine routines []routine
routinesCtx []routineCtx waitGroup sync.WaitGroup
waitGroup sync.WaitGroup lock sync.Mutex
lock sync.Mutex baseCtx context.Context
baseCtx context.Context baseCancel context.CancelFunc
baseCancel context.CancelFunc ctx context.Context
ctx context.Context cancel context.CancelFunc
cancel context.CancelFunc
} }
// NewPool creates a Pool // NewPool creates a Pool
@ -46,17 +45,9 @@ func (p *Pool) Ctx() context.Context {
return p.baseCtx return p.baseCtx
} }
// AddGoCtx adds a recoverable goroutine with a context without starting it
func (p *Pool) AddGoCtx(goroutine routineCtx) {
p.lock.Lock()
p.routinesCtx = append(p.routinesCtx, goroutine)
p.lock.Unlock()
}
// GoCtx starts a recoverable goroutine with a context // GoCtx starts a recoverable goroutine with a context
func (p *Pool) GoCtx(goroutine routineCtx) { func (p *Pool) GoCtx(goroutine routineCtx) {
p.lock.Lock() p.lock.Lock()
p.routinesCtx = append(p.routinesCtx, goroutine)
p.waitGroup.Add(1) p.waitGroup.Add(1)
Go(func() { Go(func() {
defer p.waitGroup.Done() defer p.waitGroup.Done()
@ -65,17 +56,6 @@ func (p *Pool) GoCtx(goroutine routineCtx) {
p.lock.Unlock() p.lock.Unlock()
} }
// addGo adds a recoverable goroutine, and can be stopped with stop chan
func (p *Pool) addGo(goroutine func(stop chan bool)) {
p.lock.Lock()
newRoutine := routine{
goroutine: goroutine,
stop: make(chan bool, 1),
}
p.routines = append(p.routines, newRoutine)
p.lock.Unlock()
}
// Go starts a recoverable goroutine, and can be stopped with stop chan // Go starts a recoverable goroutine, and can be stopped with stop chan
func (p *Pool) Go(goroutine func(stop chan bool)) { func (p *Pool) Go(goroutine func(stop chan bool)) {
p.lock.Lock() p.lock.Lock()
@ -114,29 +94,6 @@ func (p *Pool) Cleanup() {
p.baseCancel() p.baseCancel()
} }
// Start starts all stopped routines
func (p *Pool) Start() {
p.lock.Lock()
defer p.lock.Unlock()
p.ctx, p.cancel = context.WithCancel(p.baseCtx)
for i := range p.routines {
p.waitGroup.Add(1)
p.routines[i].stop = make(chan bool, 1)
Go(func() {
defer p.waitGroup.Done()
p.routines[i].goroutine(p.routines[i].stop)
})
}
for _, routine := range p.routinesCtx {
p.waitGroup.Add(1)
Go(func() {
defer p.waitGroup.Done()
routine(p.ctx)
})
}
}
// Go starts a recoverable goroutine // Go starts a recoverable goroutine
func Go(goroutine func()) { func Go(goroutine func()) {
GoWithRecover(goroutine, defaultRecoverGoroutine) GoWithRecover(goroutine, defaultRecoverGoroutine)

View file

@ -67,13 +67,6 @@ func TestPoolWithCtx(t *testing.T) {
p.GoCtx(testRoutine.routineCtx) p.GoCtx(testRoutine.routineCtx)
}, },
}, },
{
desc: "AddGoCtx()",
fn: func(p *Pool) {
p.AddGoCtx(testRoutine.routineCtx)
p.Start()
},
},
} }
for _, test := range testCases { for _, test := range testCases {
@ -87,9 +80,6 @@ func TestPoolWithCtx(t *testing.T) {
test.fn(p) test.fn(p)
defer p.Cleanup() defer p.Cleanup()
if len(p.routinesCtx) != 1 {
t.Fatalf("After %s, Pool did have %d goroutineCtxs, expected 1", test.desc, len(p.routinesCtx))
}
testDone := make(chan bool, 1) testDone := make(chan bool, 1)
go func() { go func() {
@ -140,40 +130,6 @@ func TestPoolWithStopChan(t *testing.T) {
} }
} }
func TestPoolStartWithStopChan(t *testing.T) {
testRoutine := newFakeRoutine()
p := NewPool(context.Background())
timer := time.NewTimer(500 * time.Millisecond)
defer timer.Stop()
// Insert the stopped test goroutine via private fields into the Pool.
// There currently is no way to insert a routine via exported funcs that is not started immediately.
p.lock.Lock()
newRoutine := routine{
goroutine: testRoutine.routine,
}
p.routines = append(p.routines, newRoutine)
p.lock.Unlock()
p.Start()
testDone := make(chan bool, 1)
go func() {
<-testRoutine.startSig
p.Cleanup()
testDone <- true
}()
select {
case <-timer.C:
testRoutine.Lock()
defer testRoutine.Unlock()
t.Fatalf("Pool.Start() did not complete in time, goroutine started equals '%t'", testRoutine.started)
case <-testDone:
return
}
}
func TestPoolCleanupWithGoPanicking(t *testing.T) { func TestPoolCleanupWithGoPanicking(t *testing.T) {
testRoutine := func(stop chan bool) { testRoutine := func(stop chan bool) {
panic("BOOM") panic("BOOM")
@ -193,26 +149,12 @@ func TestPoolCleanupWithGoPanicking(t *testing.T) {
p.Go(testRoutine) p.Go(testRoutine)
}, },
}, },
{
desc: "addGo() and Start()",
fn: func(p *Pool) {
p.addGo(testRoutine)
p.Start()
},
},
{ {
desc: "GoCtx()", desc: "GoCtx()",
fn: func(p *Pool) { fn: func(p *Pool) {
p.GoCtx(testCtxRoutine) p.GoCtx(testCtxRoutine)
}, },
}, },
{
desc: "AddGoCtx() and Start()",
fn: func(p *Pool) {
p.AddGoCtx(testCtxRoutine)
p.Start()
},
},
} }
for _, test := range testCases { for _, test := range testCases {

View file

@ -100,7 +100,7 @@ type blackholeResponseWriter struct{}
func (b blackholeResponseWriter) Flush() {} func (b blackholeResponseWriter) Flush() {}
func (b blackholeResponseWriter) Hijack() (net.Conn, *bufio.ReadWriter, error) { func (b blackholeResponseWriter) Hijack() (net.Conn, *bufio.ReadWriter, error) {
return nil, nil, errors.New("you can hijack connection on blackholeResponseWriter") return nil, nil, errors.New("connection on blackholeResponseWriter cannot be hijacked")
} }
func (b blackholeResponseWriter) Header() http.Header { func (b blackholeResponseWriter) Header() http.Header {