This repository was archived by the owner on Sep 30, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
/
Copy pathanalytics.go
117 lines (99 loc) · 3.1 KB
/
analytics.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
package main
import (
"context"
"fmt"
"runtime"
"strings"
"github.com/urfave/cli/v3"
"go.opentelemetry.io/otel/attribute"
"go.opentelemetry.io/otel/trace"
"github.com/sourcegraph/sourcegraph/dev/sg/internal/analytics"
"github.com/sourcegraph/sourcegraph/dev/sg/internal/std"
"github.com/sourcegraph/sourcegraph/dev/sg/interrupt"
)
// addAnalyticsHooks wraps command actions with analytics hooks. We reconstruct commandPath
// ourselves because the library's state (and hence .FullName()) seems to get a bit funky.
//
// It also handles watching for panics and formatting them in a useful manner.
func addAnalyticsHooks(commandPath []string, commands []*cli.Command) {
for _, command := range commands {
fullCommandPath := append(commandPath, command.Name)
if len(command.Commands) > 0 {
addAnalyticsHooks(fullCommandPath, command.Commands)
}
// No action to perform analytics on
if command.Action == nil {
continue
}
// Set up analytics hook for command
fullCommand := strings.Join(fullCommandPath, " ")
// Wrap action with analytics
wrappedAction := command.Action
command.Action = func(ctx context.Context, cmd *cli.Command) (actionErr error) {
var span *analytics.Span
ctx, span = analytics.StartSpan(ctx, fullCommand, "action",
trace.WithAttributes(
attribute.StringSlice("flags", cmd.FlagNames()),
attribute.Int("args", cmd.NArg()),
))
defer span.End()
// Make sure analytics are persisted before exit (interrupts or panics)
defer func() {
if p := recover(); p != nil {
// Render a more elegant message
std.Out.WriteWarningf("Encountered panic - please open an issue with the command output:\n\t%s",
sgBugReportTemplate)
message := fmt.Sprintf("%v:\n%s", p, getRelevantStack("addAnalyticsHooks"))
actionErr = cli.Exit(message, 1)
// Log event
span.RecordError("panic", actionErr)
}
}()
interrupt.Register(func() {
span.Cancelled()
span.End()
})
// Call the underlying action
actionErr = wrappedAction(ctx, cmd)
// Capture analytics post-run
if actionErr != nil {
span.RecordError("error", actionErr)
} else {
span.Succeeded()
}
return actionErr
}
}
}
// getRelevantStack generates a stacktrace that encapsulates the relevant parts of a
// stacktrace for user-friendly reading.
func getRelevantStack(excludeFunctions ...string) string {
callers := make([]uintptr, 32)
n := runtime.Callers(3, callers) // recover -> getRelevantStack -> runtime.Callers
frames := runtime.CallersFrames(callers[:n])
var stack strings.Builder
for {
frame, next := frames.Next()
var excludedFunction bool
for _, e := range excludeFunctions {
if strings.Contains(frame.Function, e) {
excludedFunction = true
break
}
}
// Only include frames from sg and things that are not excluded.
if !strings.Contains(frame.File, "dev/sg/") || excludedFunction {
if !next {
break
}
continue
}
stack.WriteString(frame.Function)
stack.WriteByte('\n')
stack.WriteString(fmt.Sprintf("\t%s:%d\n", frame.File, frame.Line))
if !next {
break
}
}
return stack.String()
}