-
Notifications
You must be signed in to change notification settings - Fork 577
/
events.go
168 lines (143 loc) · 5.86 KB
/
events.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
/*
Copyright 2019 The Knative Authors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package events
import (
"context"
"fmt"
"strings"
"time"
cloudevents "github.com/cloudevents/sdk-go/v2"
ceobs "github.com/cloudevents/sdk-go/v2/observability"
"go.opentelemetry.io/otel/trace"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/meta"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
kncloudevents "knative.dev/eventing/pkg/adapter/v2"
sources "knative.dev/eventing/pkg/apis/sources"
"knative.dev/eventing/pkg/observability"
)
const (
resourceGroup = "apiserversources.sources.knative.dev"
)
// MakeAddEvent returns a cloudevent when a k8s api event is created.
func MakeAddEvent(source string, apiServerSourceName string, obj interface{}, ref bool) (context.Context, cloudevents.Event, error) {
if obj == nil {
return nil, cloudevents.Event{}, fmt.Errorf("resource can not be nil")
}
object := obj.(*unstructured.Unstructured)
var data interface{}
var eventType string
if ref {
data = getRef(object)
eventType = sources.ApiServerSourceAddRefEventType
} else {
data = object
eventType = sources.ApiServerSourceAddEventType
}
return makeEvent(source, apiServerSourceName, eventType, object, data)
}
// MakeUpdateEvent returns a cloudevent when a k8s api event is updated.
func MakeUpdateEvent(source string, apiServerSourceName string, obj interface{}, ref bool) (context.Context, cloudevents.Event, error) {
if obj == nil {
return nil, cloudevents.Event{}, fmt.Errorf("resource can not be nil")
}
object := obj.(*unstructured.Unstructured)
var data interface{}
var eventType string
if ref {
data = getRef(object)
eventType = sources.ApiServerSourceUpdateRefEventType
} else {
data = object
eventType = sources.ApiServerSourceUpdateEventType
}
return makeEvent(source, apiServerSourceName, eventType, object, data)
}
// MakeDeleteEvent returns a cloudevent when a k8s api event is deleted.
func MakeDeleteEvent(source string, apiServerSourceName string, obj interface{}, ref bool) (context.Context, cloudevents.Event, error) {
if obj == nil {
return nil, cloudevents.Event{}, fmt.Errorf("resource can not be nil")
}
object := obj.(*unstructured.Unstructured)
var data interface{}
var eventType string
if ref {
data = getRef(object)
eventType = sources.ApiServerSourceDeleteRefEventType
} else {
data = object
eventType = sources.ApiServerSourceDeleteEventType
}
return makeEvent(source, apiServerSourceName, eventType, object, data)
}
func getRef(object *unstructured.Unstructured) corev1.ObjectReference {
return corev1.ObjectReference{
APIVersion: object.GetAPIVersion(),
Kind: object.GetKind(),
Name: object.GetName(),
Namespace: object.GetNamespace(),
}
}
func makeEvent(source, apiServerSourceName, eventType string, obj *unstructured.Unstructured, data interface{}) (context.Context, cloudevents.Event, error) {
resourceName := obj.GetName()
kind := obj.GetKind()
namespace := obj.GetNamespace()
subject := createSelfLink(corev1.ObjectReference{
APIVersion: obj.GetAPIVersion(),
Kind: kind,
Name: resourceName,
Namespace: namespace,
})
event := cloudevents.NewEvent(cloudevents.VersionV1)
event.SetType(eventType)
event.SetSource(source)
event.SetSubject(subject)
// We copy the resource kind, name and namespace as extensions so that triggers can do the filter based on these attributes
event.SetExtension("kind", kind)
event.SetExtension("apiversion", obj.GetAPIVersion())
event.SetExtension("name", resourceName)
event.SetExtension("namespace", namespace)
if err := event.SetData(cloudevents.ApplicationJSON, data); err != nil {
return nil, event, err
}
ctx := context.Background()
metricTag := &kncloudevents.MetricTag{
Namespace: namespace,
Name: apiServerSourceName,
ResourceGroup: resourceGroup,
}
spanName := ceobs.ClientSpanName + " process"
ctx = observability.WithSpanData(ctx, spanName, int(trace.SpanKindProducer),
observability.K8sAttributes(apiServerSourceName, namespace, resourceGroup))
ctx = kncloudevents.ContextWithMetricTag(ctx, metricTag)
ctx = cloudevents.ContextWithRetriesExponentialBackoff(ctx, 50*time.Millisecond, 5)
return ctx, event, nil
}
// Creates a URI of the form found in object metadata selfLinks
// Format looks like: /apis/feeds.knative.dev/v1alpha1/namespaces/default/feeds/k8s-events-example
// KNOWN ISSUES:
// * ObjectReference.APIVersion has no version information (e.g. serving.knative.dev rather than serving.knative.dev/v1alpha1)
// * ObjectReference does not have enough information to create the pluaralized list type (e.g. "revisions" from kind: Revision)
//
// Track these issues at https://github.com/kubernetes/kubernetes/issues/66313
// We could possibly work around this by adding a lister for the resources referenced by these events.
func createSelfLink(o corev1.ObjectReference) string {
gvr, _ := meta.UnsafeGuessKindToResource(o.GroupVersionKind())
versionNameHack := o.APIVersion
// Core API types don't have a separate package name and only have a version string (e.g. /apis/v1/namespaces/default/pods/myPod)
// To avoid weird looking strings like "v1/versionUnknown" we'll sniff for a "." in the version
if strings.Contains(versionNameHack, ".") && !strings.Contains(versionNameHack, "/") {
versionNameHack = versionNameHack + "/versionUnknown"
}
return fmt.Sprintf("/apis/%s/namespaces/%s/%s/%s", versionNameHack, o.Namespace, gvr.Resource, o.Name)
}