/
stacktrace.go
98 lines (84 loc) · 2.36 KB
/
stacktrace.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
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
// Copyright 2020 New Relic Corporation. All rights reserved.
// SPDX-License-Identifier: Apache-2.0
package internal
import (
"bytes"
"path"
"runtime"
"strings"
)
// StackTrace is a stack trace.
type StackTrace []uintptr
// GetStackTrace returns a new StackTrace.
func GetStackTrace() StackTrace {
skip := 1 // skip runtime.Callers
callers := make([]uintptr, maxStackTraceFrames)
written := runtime.Callers(skip, callers)
return callers[:written]
}
type stacktraceFrame struct {
Name string
File string
Line int64
}
func (f stacktraceFrame) formattedName() string {
if strings.HasPrefix(f.Name, "go.") {
// This indicates an anonymous struct. eg.
// "go.(*struct { github.com/newrelic/go-agent.threadWithExtras }).NoticeError"
return f.Name
}
return path.Base(f.Name)
}
func (f stacktraceFrame) isAgent() bool {
// Note this is not a contains conditional rather than a prefix
// conditional to handle anonymous functions like:
// "go.(*struct { github.com/newrelic/go-agent.threadWithExtras }).NoticeError"
return strings.Contains(f.Name, "github.com/newrelic/go-agent/internal.") ||
strings.Contains(f.Name, "github.com/newrelic/go-agent.")
}
func (f stacktraceFrame) WriteJSON(buf *bytes.Buffer) {
buf.WriteByte('{')
w := jsonFieldsWriter{buf: buf}
if f.Name != "" {
w.stringField("name", f.formattedName())
}
if f.File != "" {
w.stringField("filepath", f.File)
}
if f.Line != 0 {
w.intField("line", f.Line)
}
buf.WriteByte('}')
}
func writeFrames(buf *bytes.Buffer, frames []stacktraceFrame) {
// Remove top agent frames.
for len(frames) > 0 && frames[0].isAgent() {
frames = frames[1:]
}
// Truncate excessively long stack traces (they may be provided by the
// customer).
if len(frames) > maxStackTraceFrames {
frames = frames[0:maxStackTraceFrames]
}
buf.WriteByte('[')
for idx, frame := range frames {
if idx > 0 {
buf.WriteByte(',')
}
frame.WriteJSON(buf)
}
buf.WriteByte(']')
}
// WriteJSON adds the stack trace to the buffer in the JSON form expected by the
// collector.
func (st StackTrace) WriteJSON(buf *bytes.Buffer) {
frames := st.frames()
writeFrames(buf, frames)
}
// MarshalJSON prepares JSON in the format expected by the collector.
func (st StackTrace) MarshalJSON() ([]byte, error) {
estimate := 256 * len(st)
buf := bytes.NewBuffer(make([]byte, 0, estimate))
st.WriteJSON(buf)
return buf.Bytes(), nil
}