forked from getsentry/sentry-kubernetes
-
Notifications
You must be signed in to change notification settings - Fork 0
/
sentry_dsn_data.go
203 lines (184 loc) · 5.37 KB
/
sentry_dsn_data.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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
package main
import (
"context"
"errors"
"os"
"sync"
"github.com/getsentry/sentry-go"
v1 "k8s.io/api/apps/v1"
batchv1 "k8s.io/api/batch/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
var DSNAnnotation = "k8s.sentry.io/dsn"
var dsnClientMapping = NewDsnClientMapping()
// Map from Sentry DSN to Client
type DsnClientMapping struct {
mutex sync.RWMutex
clientMap map[string]*sentry.Client
customDsnFlag bool
}
func NewDsnClientMapping() *DsnClientMapping {
return &DsnClientMapping{
mutex: sync.RWMutex{},
clientMap: make(map[string]*sentry.Client),
customDsnFlag: isTruthy(os.Getenv("SENTRY_K8S_CUSTOM_DSNS")),
}
}
// Return client if added successfully
// (also returns client if already exists)
func (d *DsnClientMapping) AddClientToMap(options sentry.ClientOptions) (*sentry.Client, error) {
d.mutex.Lock()
defer d.mutex.Unlock()
// Create a new client for the dsn
// even if client already exists, it
// will be re-initialized with a new client
newClient, err := sentry.NewClient(
sentry.ClientOptions{
Dsn: options.Dsn,
Debug: true,
AttachStacktrace: true,
},
)
if err != nil {
return nil, err
}
d.clientMap[options.Dsn] = newClient
return newClient, nil
}
// Retrieve a client with given dsn
func (d *DsnClientMapping) GetClientFromMap(dsn string) (*sentry.Client, bool) {
d.mutex.RLock()
defer d.mutex.RUnlock()
// Check if we have this dsn
existingClient, ok := d.clientMap[dsn]
return existingClient, ok
}
func (d *DsnClientMapping) GetClientFromObject(ctx context.Context, object metav1.Object, clientOptions sentry.ClientOptions) (*sentry.Client, bool) {
// If the custom DSN flag is set to false
// then avoid searching for the custom DSN
// or adding an alternative client and instead
// just return nil as the client
if !d.customDsnFlag {
return nil, false
}
// Find DSN annotation from the object
altDsn, err := searchDsn(ctx, object)
if err != nil {
return nil, false
}
// If we did find an alternative DSN
if altDsn != "" {
// Attempt to retrieve the corresponding client
client, _ := dsnClientMapping.GetClientFromMap(altDsn)
if client == nil {
// create new client
clientOptions.Dsn = altDsn
client, err = dsnClientMapping.AddClientToMap(clientOptions)
if err != nil {
return nil, false
}
}
return client, true
}
return nil, false
}
// Recursive function to find if there is a DSN annotation
func searchDsn(ctx context.Context, obj metav1.Object) (string, error) {
dsn, ok := obj.GetAnnotations()[DSNAnnotation]
if ok {
return dsn, nil
}
if len(obj.GetOwnerReferences()) == 0 {
return "", nil
}
owningRef := obj.GetOwnerReferences()[0]
owningObject, ok := findObject(ctx, owningRef.Kind, obj.GetNamespace(), owningRef.Name)
if !ok {
return "", errors.New("the DSN cannot be found")
}
return searchDsn(ctx, owningObject)
}
func findObject(ctx context.Context, kind string, namespace string, name string) (metav1.Object, bool) {
clientset, err := getClientsetFromContext(ctx)
if err != nil {
return nil, false
}
switch kind {
case KindPod:
pod, err := clientset.CoreV1().Pods(namespace).Get(context.Background(), name, metav1.GetOptions{})
if err != nil {
return nil, false
}
return pod, true
case KindReplicaset:
var replicaSet *v1.ReplicaSet
// Check if the replicaset is available in the indexer first
if replicasetInformer != nil {
obj, ok, err := replicasetInformer.GetIndexer().GetByKey(namespace + "/" + name)
if ok && err == nil {
replicaSet = obj.(*v1.ReplicaSet)
}
}
if replicaSet == nil {
// Query replicaset with kubernetes API
replicaSet, err = clientset.AppsV1().ReplicaSets(namespace).Get(context.Background(), name, metav1.GetOptions{})
if err != nil {
return nil, false
}
}
return replicaSet, true
case KindDeployment:
var deployment *v1.Deployment
// Check if the deployment is available in the indexer first
if deploymentInformer != nil {
obj, ok, err := deploymentInformer.GetIndexer().GetByKey(namespace + "/" + name)
if ok && err == nil {
deployment = obj.(*v1.Deployment)
}
}
if deployment == nil {
// Query deployment with kubernetes API
deployment, err = clientset.AppsV1().Deployments(namespace).Get(context.Background(), name, metav1.GetOptions{})
if err != nil {
return nil, false
}
}
return deployment, true
case KindJob:
var job *batchv1.Job
// Check if the job is available in the indexer first
if jobInformer != nil {
obj, ok, err := jobInformer.GetIndexer().GetByKey(namespace + "/" + name)
if ok && err == nil {
job = obj.(*batchv1.Job)
}
}
if job == nil {
// Query job with kubernetes API
job, err = clientset.BatchV1().Jobs(namespace).Get(context.Background(), name, metav1.GetOptions{})
if err != nil {
return nil, false
}
}
return job, true
case KindCronjob:
var cronjob *batchv1.CronJob
// Check if the cronjob is available in the indexer first
if cronjobInformer != nil {
obj, ok, err := cronjobInformer.GetIndexer().GetByKey(namespace + "/" + name)
if ok && err == nil {
cronjob = obj.(*batchv1.CronJob)
}
}
if cronjob == nil {
// Query cronjob with kubernetes API
cronjob, err = clientset.BatchV1().CronJobs(namespace).Get(context.Background(), name, metav1.GetOptions{})
if err != nil {
return nil, false
}
}
return cronjob, true
default:
return nil, false
}
}