Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>main
parent
b0c10fa610
commit
703d789005
@ -0,0 +1,45 @@ |
||||
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
// Package logheap logs a heap pprof profile.
|
||||
package logheap |
||||
|
||||
import ( |
||||
"bytes" |
||||
"encoding/json" |
||||
"io" |
||||
"os" |
||||
"runtime" |
||||
"runtime/pprof" |
||||
"time" |
||||
) |
||||
|
||||
// LogHeap writes a JSON logtail record with the base64 heap pprof to
|
||||
// os.Stderr.
|
||||
func LogHeap() { |
||||
logHeap(os.Stderr) |
||||
} |
||||
|
||||
type logTail struct { |
||||
ClientTime string `json:"client_time"` |
||||
} |
||||
|
||||
type pprofRec struct { |
||||
Heap []byte `json:"heap,omitempty"` |
||||
} |
||||
|
||||
type logLine struct { |
||||
LogTail logTail `json:"logtail"` |
||||
Pprof pprofRec `json:"pprof"` |
||||
} |
||||
|
||||
func logHeap(w io.Writer) error { |
||||
runtime.GC() |
||||
buf := new(bytes.Buffer) |
||||
pprof.WriteHeapProfile(buf) |
||||
return json.NewEncoder(w).Encode(logLine{ |
||||
LogTail: logTail{ClientTime: time.Now().Format(time.RFC3339Nano)}, |
||||
Pprof: pprofRec{Heap: buf.Bytes()}, |
||||
}) |
||||
} |
||||
@ -0,0 +1,40 @@ |
||||
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
package logheap |
||||
|
||||
import ( |
||||
"bytes" |
||||
"compress/gzip" |
||||
"encoding/json" |
||||
"io/ioutil" |
||||
"testing" |
||||
) |
||||
|
||||
func TestLogHeap(t *testing.T) { |
||||
var buf bytes.Buffer |
||||
if err := logHeap(&buf); err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
t.Logf("Got line: %s", buf.Bytes()) |
||||
|
||||
var ll logLine |
||||
if err := json.Unmarshal(buf.Bytes(), &ll); err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
|
||||
zr, err := gzip.NewReader(bytes.NewReader(ll.Pprof.Heap)) |
||||
if err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
rawProto, err := ioutil.ReadAll(zr) |
||||
if err != nil { |
||||
t.Fatal(err) |
||||
} |
||||
// Just sanity check it. Too lazy to properly decode the protobuf. But see that
|
||||
// it contains an expected sample name.
|
||||
if !bytes.Contains(rawProto, []byte("alloc_objects")) { |
||||
t.Errorf("raw proto didn't contain `alloc_objects`: %q", rawProto) |
||||
} |
||||
} |
||||
Loading…
Reference in new issue