-
Notifications
You must be signed in to change notification settings - Fork 21
/
api.go
117 lines (101 loc) · 3.39 KB
/
api.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 pyroscope
import (
"context"
"fmt"
"os"
"runtime/pprof"
"time"
"github.com/grafana/pyroscope-go/upstream/remote"
)
type Config struct {
ApplicationName string // e.g backend.purchases
Tags map[string]string
ServerAddress string // e.g http://pyroscope.services.internal:4040
BasicAuthUser string // http basic auth user
BasicAuthPassword string // http basic auth password
TenantID string // specify TenantId when using phlare multi-tenancy
UploadRate time.Duration
Logger Logger
ProfileTypes []ProfileType
DisableGCRuns bool // this will disable automatic runtime.GC runs between getting the heap profiles
HTTPHeaders map[string]string
// Deprecated: the field will be removed in future releases.
// Use BasicAuthUser and BasicAuthPassword instead.
AuthToken string // specify this token when using pyroscope cloud
// Deprecated: the field will be removed in future releases.
// Use UploadRate instead.
DisableAutomaticResets bool
// Deprecated: the field will be removed in future releases.
// DisableCumulativeMerge is ignored.
DisableCumulativeMerge bool
// Deprecated: the field will be removed in future releases.
// SampleRate is set to 100 and is not configurable.
SampleRate uint32
}
type Profiler struct {
session *Session
uploader *remote.Remote
}
// Start starts continuously profiling go code
func Start(cfg Config) (*Profiler, error) {
if len(cfg.ProfileTypes) == 0 {
cfg.ProfileTypes = DefaultProfileTypes
}
if cfg.Logger == nil {
cfg.Logger = noopLogger
}
// Override the address to use when the environment variable is defined.
// This is useful to support adhoc push ingestion.
if address, ok := os.LookupEnv("PYROSCOPE_ADHOC_SERVER_ADDRESS"); ok {
cfg.ServerAddress = address
}
rc := remote.Config{
AuthToken: cfg.AuthToken,
TenantID: cfg.TenantID,
BasicAuthUser: cfg.BasicAuthUser,
BasicAuthPassword: cfg.BasicAuthPassword,
HTTPHeaders: cfg.HTTPHeaders,
Address: cfg.ServerAddress,
Threads: 5, // per each profile type upload
Timeout: 30 * time.Second,
Logger: cfg.Logger,
}
uploader, err := remote.NewRemote(rc)
if err != nil {
return nil, err
}
sc := SessionConfig{
Upstream: uploader,
Logger: cfg.Logger,
AppName: cfg.ApplicationName,
Tags: cfg.Tags,
ProfilingTypes: cfg.ProfileTypes,
DisableGCRuns: cfg.DisableGCRuns,
DisableAutomaticResets: cfg.DisableAutomaticResets,
UploadRate: cfg.UploadRate,
}
s, err := NewSession(sc)
if err != nil {
return nil, fmt.Errorf("new session: %w", err)
}
uploader.Start()
if err = s.Start(); err != nil {
return nil, fmt.Errorf("start session: %w", err)
}
return &Profiler{session: s, uploader: uploader}, nil
}
// Stop stops continuous profiling session and uploads the remaining profiling data
func (p *Profiler) Stop() error {
p.session.Stop()
p.uploader.Stop()
return nil
}
// Flush resets current profiling session. if wait is true, also waits for all profiles to be uploaded synchronously
func (p *Profiler) Flush(wait bool) {
p.session.flush(wait)
}
type LabelSet = pprof.LabelSet
var Labels = pprof.Labels
func TagWrapper(ctx context.Context, labels LabelSet, cb func(context.Context)) {
pprof.Do(ctx, labels, func(c context.Context) { cb(c) })
}