forked from rollbar/rollbar-go
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstack.go
91 lines (81 loc) · 2.25 KB
/
stack.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
package rollbar
import (
"fmt"
"hash/crc32"
"os"
"runtime"
"strings"
)
var (
knownFilePathPatterns = []string{
"github.com/",
"code.google.com/",
"bitbucket.org/",
"launchpad.net/",
}
)
// Frame represents one frame in a stack trace
type Frame struct {
// Filename is the name of the file for this frame
Filename string `json:"filename"`
// Method is the name of the method for this frame
Method string `json:"method"`
// Line is the line number in the file for this frame
Line int `json:"lineno"`
}
// A Stack is a slice of frames
type Stack []Frame
// BuildStack uses the Go runtime to construct a slice of frames optionally skipping the number of
// frames specified by the input skip argument.
func BuildStack(skip int) Stack {
stack := make(Stack, 0)
for i := skip; ; i++ {
pc, file, line, ok := runtime.Caller(i)
if !ok {
break
}
file = shortenFilePath(file)
stack = append(stack, Frame{file, functionName(pc), line})
}
return stack
}
// Fingerprint creates a fingerprint that uniqely identify a given message. We use the full
// callstack, including file names. That ensure that there are no false duplicates
// but also means that after changing the code (adding/removing lines), the
// fingerprints will change. It's a trade-off.
func (s Stack) Fingerprint() string {
hash := crc32.NewIEEE()
for _, frame := range s {
fmt.Fprintf(hash, "%s%s%d", frame.Filename, frame.Method, frame.Line)
}
return fmt.Sprintf("%x", hash.Sum32())
}
// Remove un-needed information from the source file path. This makes them
// shorter in Rollbar UI as well as making them the same, regardless of the
// machine the code was compiled on.
//
// Examples:
// /usr/local/go/src/pkg/runtime/proc.c -> pkg/runtime/proc.c
// /home/foo/go/src/github.com/rollbar/rollbar.go -> github.com/rollbar/rollbar.go
func shortenFilePath(s string) string {
idx := strings.Index(s, "/src/pkg/")
if idx != -1 {
return s[idx+5:]
}
for _, pattern := range knownFilePathPatterns {
idx = strings.Index(s, pattern)
if idx != -1 {
return s[idx:]
}
}
return s
}
func functionName(pc uintptr) string {
fn := runtime.FuncForPC(pc)
if fn == nil {
return "???"
}
name := fn.Name()
end := strings.LastIndex(name, string(os.PathSeparator))
return name[end+1:]
}