cmd/stcrashreceiver: Don't leak clients

Use a global raven.Client because they allocate an http.Client for each,
with a separate CA bundle and infinite connection idle time. Infinite
connection idle time means that if the client is never used again it
will always keep the connection around, not verifying whether it's
closed server side or not. This leaks about a megabyte of memory for
each client every created.

client.Close() doesn't help with this because the http.Client is still
around, retained by its own goroutines.

The thing with the map is just to retain the API on sendReport, even
though there will in practice only ever be one DSN per process
instance...
This commit is contained in:
Jakob Borg 2019-07-09 11:08:59 +02:00
parent 4cba433852
commit d0ab65a178

View File

@ -12,6 +12,7 @@ import (
"io/ioutil" "io/ioutil"
"regexp" "regexp"
"strings" "strings"
"sync"
raven "github.com/getsentry/raven-go" raven "github.com/getsentry/raven-go"
"github.com/maruel/panicparse/stack" "github.com/maruel/panicparse/stack"
@ -25,16 +26,28 @@ func init() {
raven.SetSourceCodeLoader(loader) raven.SetSourceCodeLoader(loader)
} }
var (
clients = make(map[string]*raven.Client)
clientsMut sync.Mutex
)
func sendReport(dsn, path string, report []byte) error { func sendReport(dsn, path string, report []byte) error {
pkt, err := parseReport(path, report) pkt, err := parseReport(path, report)
if err != nil { if err != nil {
return err return err
} }
cli, err := raven.New(dsn) clientsMut.Lock()
defer clientsMut.Unlock()
cli, ok := clients[dsn]
if !ok {
cli, err = raven.New(dsn)
if err != nil { if err != nil {
return err return err
} }
clients[dsn] = cli
}
// The client sets release and such on the packet before sending, in the // The client sets release and such on the packet before sending, in the
// misguided idea that it knows this better than than the packet we give // misguided idea that it knows this better than than the packet we give
@ -42,6 +55,7 @@ func sendReport(dsn, path string, report []byte) error {
cli.SetRelease(pkt.Release) cli.SetRelease(pkt.Release)
cli.SetEnvironment(pkt.Environment) cli.SetEnvironment(pkt.Environment)
defer cli.Wait()
_, errC := cli.Capture(pkt, nil) _, errC := cli.Capture(pkt, nil)
return <-errC return <-errC
} }