-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathstack2struct.go
125 lines (106 loc) · 3.65 KB
/
stack2struct.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
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
// This parses raw golang stack traces ([]byte) to a slice of well formated structs.
// As this package will need to evolve with the development of go's stack trace
// format, this is the stack format the package currently works with:
//
// 1: goroutine x [running]: <-- ignore this line
// 2: path/to/package.functionName()
// 3: path/to/responsible/file:lineNumber +0xdeadbeef <-- memory address optional
// ... repeat 2 + 3 for each stack trace element
//
// To work with this, you need a type satisfying the interface
//
// type stackTrace interface {
// AddEntry(lineNumber int, packageName string, fileName string, methodName string)
// }
//
// and from there you can do whatever you like with the accumulated data.
package raygun4go
import (
"fmt"
"runtime"
"strconv"
"strings"
goerrors "github.com/go-errors/errors"
)
// stackTrace is the interface a target stack has to satisfy.
type stackTrace interface {
AddEntry(lineNumber int, packageName string, fileName string, methodName string)
}
// Current loads the current stacktrace into a given stack
func Current(stack stackTrace) {
rawStack := make([]byte, 1<<16)
rawStack = rawStack[:runtime.Stack(rawStack, false)]
Parse(rawStack, stack)
}
// Parse loads the stack trace (given as trace) into the given stack.
// See Current() on how to obtain a stack trace.
func Parse(trace []byte, stack stackTrace) {
lines := strings.Split(strings.ReplaceAll(string(trace), "\r\n", "\n"), "\n")
var lineNumber int
var fileName, packageName, methodName string
for index, line := range lines[1:] {
if len(line) == 0 {
continue
}
if index%2 == 0 {
packageName, methodName = extractPackageName(line)
} else {
lineNumber, fileName = extractLineNumberAndFile(line)
stack.AddEntry(lineNumber, packageName, fileName, methodName)
}
}
}
// LoadGoErrorStack loads the strack trace (given as frames) into the given stack.
func LoadGoErrorStack(frames []goerrors.StackFrame, stack stackTrace) {
for _, frame := range frames {
stack.AddEntry(frame.LineNumber, frame.Package, frame.File, frame.Name)
}
}
// extractPageName receives a trace line and extracts packageName and
// methodName.
func extractPackageName(line string) (packageName, methodName string) {
packagePath, packageNameAndFunction := splitAtLastSlash(line)
parts := strings.Split(packageNameAndFunction, ".")
if len(parts) > 1 {
packageName = parts[0]
if len(packagePath) > 0 {
packageName = fmt.Sprintf("%s/%s", packagePath, packageName)
}
methodName = strings.Join(parts[1:], ".")
} else {
methodName = parts[0]
}
return
}
// extractLineNumberAndFile receives a trace line and extracts lineNumber and
// fileName.
func extractLineNumberAndFile(line string) (lineNumber int, fileName string) {
_, fileAndLine := splitAtLastSlash(line)
fileAndLine = removeSpaceAndSuffix(fileAndLine)
parts := strings.Split(fileAndLine, ":")
lineNumber = 0
if len(parts) >= 2 {
numberAsString := parts[1]
number, _ := strconv.ParseUint(numberAsString, 10, 32)
lineNumber = int(number)
}
fileName = parts[0]
return lineNumber, fileName
}
// splitAtLastSlash splits a string at the last found slash and returns the
// respective strings left and right of the slash.
func splitAtLastSlash(line string) (left, right string) {
parts := strings.Split(line, "/")
right = parts[len(parts)-1]
left = strings.Join(parts[:len(parts)-1], "/")
return
}
// removeSpaceAndSuffix splits the given string at ' ' and cuts off the part
// found after the last space.
func removeSpaceAndSuffix(line string) string {
parts := strings.Split(line, " ")
if len(parts) <= 1 {
return line
}
return strings.Join(parts[:len(parts)-1], " ")
}