Leaky Go Channels


These simple go tests check the “leaky-ness” of using channels in Go. There are two methods described here; one using both a local context, and the parent context. When tests are run against both, the LeakyAsync method runs faster, but fails the leak checker as goroutines are not resolved or cleaned up.

In a production system with possibly thousands of go routines being spun up, this could result in massive memory leaks and a deadlock situation in the go binary.

it’s recommended to use the leakchecker library to determine if goroutines get cleaned up.

package perform

import (
	"context"
	"math"
)

func Selecting(parent chan struct{}) {
	i := 0
	for {
		// Selecting within the infinite loop provides
		// control from the parent chan.
		// If the parent closes, we we can exit the loop and do any cleanup
		select {
		case <-parent:
			return
		default:
		}

		i++

		if i == math.MaxInt32 {
			// Simulate an error that exits the process loop
			break
		}
	}
}

func LeakyAsync(parent chan struct{}) {
	// Start a go routine to read and block off the parent chan.
	// If the parent chan closes, we can clean up within the go routine
	// without having to perform a "select" on each iteration
	// However, this go routine will never be garbage collected
	// if the parent chan does not close and any subsequent cleanup will
	// be left to leak
	go func(c <-chan struct{}) {
		<-c
	}(parent)

	i := 0
	for {
		i++

		if i == math.MaxInt32 {
			// Simulate an error that exits the process loop
			break
		}
	}
}

func ContextAsync(parentCtx context.Context) {
	// Generate a child context from a passed in parent context.
	// If the parent is closed or canceled,
	// the child will also be closed.
	// We can then safely start a go routine that will block on the
	// child's Done channel yet will still continue if the parent is canceled.
	ctx, cancel := context.WithCancel(parentCtx)
	defer cancel()

	go func(ctx context.Context) {
		<-ctx.Done()
	}(ctx)

	i := 0
	for {
		i++

		if i == math.MaxInt32 {
			// Simulate an error that exits the process loop
			break
		}
	}
}
package perform

import (
	"context"
	"testing"

	"github.com/fortytw2/leaktest"
)

func TestSelecting(t *testing.T) {
	leakChecker := leaktest.Check(t)

	c := make(chan struct{}, 1)
	Selecting(c)

	leakChecker()
	c <- struct{}{}
}

func BenchmarkSelecting(b *testing.B) {
	for n := 0; n < b.N; n++ {
		c := make(chan struct{})
		Selecting(c)
	}
}

func TestLeakyAsync(t *testing.T) {
	leakChecker := leaktest.Check(t)

	c := make(chan struct{}, 1)
	LeakyAsync(c)

	leakChecker()
	c <- struct{}{}
}

func BenchmarkLeakyAsync(b *testing.B) {
	for n := 0; n < b.N; n++ {
		c := make(chan struct{})
		LeakyAsync(c)
	}
}

func TestContextAsync(t *testing.T) {
	leakChecker := leaktest.Check(t)

	ctx, cancel := context.WithCancel(context.Background())
	ContextAsync(ctx)

	leakChecker()
	cancel()
}

func BenchmarkContextAsync(b *testing.B) {
	for n := 0; n < b.N; n++ {
		ctx, _ := context.WithCancel(context.Background())
		ContextAsync(ctx)
	}
}

Run the test suite with the leakchecker library

❯ go test -v
=== RUN   TestSelecting
done checking leak
--- PASS: TestSelecting (11.30s)
=== RUN   TestLeakyAsync
    TestLeakyAsync: leaktest.go:132: leaktest: timed out checking goroutines
    TestLeakyAsync: leaktest.go:150: leaktest: leaked goroutine: goroutine 25 [chan receive]:
        perform.LeakyAsync.func1(0xc00008c1e0)
        	/Users/jmcbride/workspace/channels-testing/perform.go:37 +0x34
        created by perform.LeakyAsync
        	/Users/jmcbride/workspace/channels-testing/perform.go:36 +0x3f
--- FAIL: TestLeakyAsync (5.57s)
=== RUN   TestContextAsync
--- PASS: TestContextAsync (0.57s)

Run the benchmarks with bench and benchmem to see performance

❯ go test -v -bench=.  -benchmem -run "Bench*"
goos: darwin
goarch: amd64
pkg: perform
BenchmarkSelecting
BenchmarkSelecting-8      	       1	10114375732 ns/op	     104 B/op	       2 allocs/op
BenchmarkLeakyAsync
BenchmarkLeakyAsync-8     	       2	 585489776 ns/op	     704 B/op	       3 allocs/op
BenchmarkContextAsync
BenchmarkContextAsync-8   	       2	 570398894 ns/op	     976 B/op	       9 allocs/op
PASS
ok  	perform	13.655s

LeakyAsync is roughly 2 times faster. But fails the leak checker test as the goroutine is not resolved.

Selecting is slow because it performs a select on every iteration of the for loop.

ContextAsync is the best of both worlds. We don’t have to do a select within the for loop, yet we avoid a go routine leak.


If you found this blog post valuable, consider subscribing to future posts via RSS or buying me a coffee via GitHub sponsors.