-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathtrace.go
53 lines (46 loc) · 1.42 KB
/
trace.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
package metrics
import (
"context"
"fmt"
"os"
rtrace "runtime/trace"
"time"
"golang.org/x/exp/trace"
)
type TraceRecorder struct {
*trace.FlightRecorder
snapshotThreshold time.Duration
}
// NewTraceRecorder creates new trace flight recorder that will continuously record latest execution trace in a circullar buffer
// and snapshot it to file only if region takes more than snapshotThreshold
func NewTraceRecorder(period, snapshotThreshold time.Duration, bufferSizeBytes int) *TraceRecorder {
tr := &TraceRecorder{
FlightRecorder: trace.NewFlightRecorder(),
snapshotThreshold: snapshotThreshold,
}
tr.SetPeriod(period)
tr.SetSize(bufferSizeBytes)
return tr
}
// StartRegion starts measuring execution time of a region and if it passes the snapshotThreshold
// then it flushes recorder trace buffer to file
func (tr TraceRecorder) StartRegion(tagName, tagValue string) (stopRegion func() error) {
start := time.Now()
_, task := rtrace.NewTask(context.Background(), fmt.Sprintf("%s=%s", tagName, tagValue))
return func() error {
task.End()
if time.Since(start) > tr.snapshotThreshold { // snapshot trace
fileName := fmt.Sprintf("trace-%s-%s-%d.out", tagName, tagValue, start.Unix())
fmt.Printf("::: writing Trace Recorder snapshot to file %s :::\n", fileName)
f, err := os.Create(fileName)
if err != nil {
return err
}
_, err = tr.WriteTo(f)
if err != nil {
return err
}
}
return nil
}
}