Skip to content

Commit ad65aae

Browse files
committed
feat(alertchannels): Add webhook alertchannel type
* Add webhook alertchannel type * Add general alertchannel configurations (send failure, degraded, recovery, ssl expiry) * Update docs, examples and tests
1 parent b308dea commit ad65aae

9 files changed

+484
-43
lines changed

api/checkly/v1alpha1/alertchannel_types.go

+26-2
Original file line numberDiff line numberDiff line change
@@ -28,16 +28,28 @@ type AlertChannelSpec struct {
2828
// Important: Run "make" to regenerate code after modifying this file
2929

3030
// SendRecovery determines if the Recovery event should be sent to the alert channel
31-
SendRecovery bool `json:"sendrecovery,omitempty"`
31+
SendRecovery bool `json:"sendRecovery,omitempty"`
3232

3333
// SendFailure determines if the Failure event should be sent to the alerting channel
34-
SendFailure bool `json:"sendfailure,omitempty"`
34+
SendFailure bool `json:"sendFailure,omitempty"`
35+
36+
// SendDegraded determines if the Failure event should be sent to the alerting channel
37+
SendDegraded bool `json:"sendDegraded,omitempty"`
38+
39+
// SSLExpiry determine if alerts on SSL Expiry should be sent
40+
SSLExpiry bool `json:"sslExpiry,omitempty"`
41+
42+
// SSLExpiryThreshold At what moment in time to start alerting on SSL certificates.
43+
SSLExpiryThreshold int `json:"sslExpiryThreshold,omitempty"`
3544

3645
// OpsGenie holds information about the Opsgenie alert configuration
3746
OpsGenie AlertChannelOpsGenie `json:"opsgenie,omitempty"`
3847

3948
// Email holds information about the Email alert configuration
4049
Email checkly.AlertChannelEmail `json:"email,omitempty"`
50+
51+
// Webhook holds information about the Webhook alert configuration
52+
Webhook AlertChannelWebhook `json:"webhook,omitempty"`
4153
}
4254

4355
type AlertChannelOpsGenie struct {
@@ -51,6 +63,18 @@ type AlertChannelOpsGenie struct {
5163
Priority string `json:"priority,omitempty"`
5264
}
5365

66+
// AlertChannelWebhook is a custom struct to hold information about Webhook configuration, source https://github.com/checkly/checkly-go-sdk/blame/main/types.go#L799-L808
67+
type AlertChannelWebhook struct {
68+
Name string `json:"name"`
69+
URL string `json:"url"`
70+
WebhookType string `json:"webhookType,omitempty"`
71+
Method string `json:"method"`
72+
Template string `json:"template,omitempty"`
73+
WebhookSecret corev1.ObjectReference `json:"webhookSecret,omitempty"`
74+
Headers []checkly.KeyValue `json:"headers,omitempty"`
75+
QueryParameters []checkly.KeyValue `json:"queryParameters,omitempty"`
76+
}
77+
5478
// AlertChannelStatus defines the observed state of AlertChannel
5579
type AlertChannelStatus struct {
5680
// INSERT ADDITIONAL STATUS FIELD - define observed state of cluster

api/checkly/v1alpha1/zz_generated.deepcopy.go

+29-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

config/crd/bases/k8s.checklyhq.com_alertchannels.yaml

+129-2
Original file line numberDiff line numberDiff line change
@@ -105,14 +105,141 @@ spec:
105105
required:
106106
- apisecret
107107
type: object
108-
sendfailure:
108+
sendDegraded:
109+
description: SendDegraded determines if the Failure event should be
110+
sent to the alerting channel
111+
type: boolean
112+
sendFailure:
109113
description: SendFailure determines if the Failure event should be
110114
sent to the alerting channel
111115
type: boolean
112-
sendrecovery:
116+
sendRecovery:
113117
description: SendRecovery determines if the Recovery event should
114118
be sent to the alert channel
115119
type: boolean
120+
sslExpiry:
121+
description: SSLExpiry determine if alerts on SSL Expiry should be
122+
sent
123+
type: boolean
124+
sslExpiryThreshold:
125+
description: SSLExpiryThreshold At what moment in time to start alerting
126+
on SSL certificates.
127+
type: integer
128+
webhook:
129+
description: Webhook holds information about the Webhook alert configuration
130+
properties:
131+
headers:
132+
items:
133+
description: |-
134+
KeyValue represents a key-value pair, for example a request header setting,
135+
or a query parameter.
136+
properties:
137+
key:
138+
type: string
139+
locked:
140+
type: boolean
141+
value:
142+
type: string
143+
required:
144+
- key
145+
- locked
146+
- value
147+
type: object
148+
type: array
149+
method:
150+
type: string
151+
name:
152+
type: string
153+
queryParameters:
154+
items:
155+
description: |-
156+
KeyValue represents a key-value pair, for example a request header setting,
157+
or a query parameter.
158+
properties:
159+
key:
160+
type: string
161+
locked:
162+
type: boolean
163+
value:
164+
type: string
165+
required:
166+
- key
167+
- locked
168+
- value
169+
type: object
170+
type: array
171+
template:
172+
type: string
173+
url:
174+
type: string
175+
webhookSecret:
176+
description: |-
177+
ObjectReference contains enough information to let you inspect or modify the referred object.
178+
---
179+
New uses of this type are discouraged because of difficulty describing its usage when embedded in APIs.
180+
1. Ignored fields. It includes many fields which are not generally honored. For instance, ResourceVersion and FieldPath are both very rarely valid in actual usage.
181+
2. Invalid usage help. It is impossible to add specific help for individual usage. In most embedded usages, there are particular
182+
restrictions like, "must refer only to types A and B" or "UID not honored" or "name must be restricted".
183+
Those cannot be well described when embedded.
184+
3. Inconsistent validation. Because the usages are different, the validation rules are different by usage, which makes it hard for users to predict what will happen.
185+
4. The fields are both imprecise and overly precise. Kind is not a precise mapping to a URL. This can produce ambiguity
186+
during interpretation and require a REST mapping. In most cases, the dependency is on the group,resource tuple
187+
and the version of the actual struct is irrelevant.
188+
5. We cannot easily change it. Because this type is embedded in many locations, updates to this type
189+
will affect numerous schemas. Don't make new APIs embed an underspecified API type they do not control.
190+
191+
192+
Instead of using this type, create a locally provided and used type that is well-focused on your reference.
193+
For example, ServiceReferences for admission registration: https://github.com/kubernetes/api/blob/release-1.17/admissionregistration/v1/types.go#L533 .
194+
properties:
195+
apiVersion:
196+
description: API version of the referent.
197+
type: string
198+
fieldPath:
199+
description: |-
200+
If referring to a piece of an object instead of an entire object, this string
201+
should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2].
202+
For example, if the object reference is to a container within a pod, this would take on a value like:
203+
"spec.containers{name}" (where "name" refers to the name of the container that triggered
204+
the event) or if no container name is specified "spec.containers[2]" (container with
205+
index 2 in this pod). This syntax is chosen only to have some well-defined way of
206+
referencing a part of an object.
207+
TODO: this design is not final and this field is subject to change in the future.
208+
type: string
209+
kind:
210+
description: |-
211+
Kind of the referent.
212+
More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
213+
type: string
214+
name:
215+
description: |-
216+
Name of the referent.
217+
More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
218+
type: string
219+
namespace:
220+
description: |-
221+
Namespace of the referent.
222+
More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/
223+
type: string
224+
resourceVersion:
225+
description: |-
226+
Specific resourceVersion to which this reference is made, if any.
227+
More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency
228+
type: string
229+
uid:
230+
description: |-
231+
UID of the referent.
232+
More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids
233+
type: string
234+
type: object
235+
x-kubernetes-map-type: atomic
236+
webhookType:
237+
type: string
238+
required:
239+
- method
240+
- name
241+
- url
242+
type: object
116243
type: object
117244
status:
118245
description: AlertChannelStatus defines the observed state of AlertChannel

config/samples/checkly_v1alpha1_alertchannel.yaml

+26
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,11 @@ kind: AlertChannel
33
metadata:
44
name: alertchannel-sample
55
spec:
6+
sendRecovery: false
7+
sendFailure: true
8+
sendDegraded: true
9+
sslExpiry: true
10+
sslExpiryThreshold: 30
611
# only one of the below can be specified at once, either email or opsgenie
712
email:
813
address: "[email protected]"
@@ -13,3 +18,24 @@ spec:
1318
# fieldPath: "TEST" # Key inside the secret
1419
# priority: "P3"
1520
# region: "US"
21+
# sslExpiry: true
22+
# sendDegraded: true
23+
# sslExpiryThreshold: 15
24+
# webhook:
25+
# name: foo # Name of the webhook
26+
# url: https://foo.bar # URL of the webhook
27+
# webhookType : baz # Type of webhook
28+
# method: POST # Method of webhook
29+
# template: testing # Checkly webhook template
30+
# webhookSecret:
31+
# name: test-secret # Name of the secret or configmap which holds the webhook secret
32+
# namespace: default # Namespace of the secret or configmap
33+
# fieldPath: "SECRET_KEY" # Key inside the secret or configmap
34+
# headers:
35+
# - key: "foo"
36+
# value: "bar"
37+
# locked: true
38+
# queryParameters:
39+
# - key: "bar"
40+
# value: "baz"
41+
# locked: false

docs/alert-channels.md

+78-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,36 @@ See the [official checkly docs](https://www.checklyhq.com/docs/alerting/) on wha
66

77
The name of the Alert channel derives from the `metadata.name` of the created kubernetes resource.
88

9-
We're supporting the email and OpsGenie configurations. You can not specify both in a config as each alert channel can only have one channel, if you want to alert to multiple channels, create a resource for each and later reference them in the check group configuration.
9+
We're supporting the email, OpsGenie and webhook configurations. You can not specify all in a config as each alert channel can only have one channel, if you want to alert to multiple channels, create a resource for each and later reference them in the check group configuration.
10+
11+
### Spec
12+
13+
| Option | Details | Default |
14+
|--------------|-----------|------------|
15+
| `sendRecovery` | Bool; Should recovery alerts be sent | none |
16+
| `sendFailure` | Bool; Should failure alerts be sent | none |
17+
| `sslExpiry` | Bool; Should ssl expiry check alerts be sent | none |
18+
| `sendDegraded` | Bool; Should degraded alerts be sent | none |
19+
| `sslExpiryThreshold` | int; At what moment in time to start alerting on SSL certificates. | none |
20+
| `email.address` | string; Which email address should the alert be sent to | none |
21+
| `opsgenie.apikey.name` | string; Name of the secret or configmap which holds the Opsgenie API key | none |
22+
| `opsgenie.apikey.namespace` | string; Namespace of the secret or configmap | none |
23+
| `opsgenie.apikey.fieldPath` | string; Key inside the secret or configmap | none |
24+
| `webhook.name` | string; Name for the webhook | none |
25+
| `webhook.url` | string; URL for the webhook | none |
26+
| `webhook.webhookType` | string; TODO: can't determine what this is | none |
27+
| `webhook.method` | string; HTTP type for the webhook (POST/GET/PUT/HEAD/DELETE/PATCH) | none |
28+
| `webhook.template` | string; Template for webhook message | none |
29+
| `webhook.name` | string; Name for the webhook | none |
30+
| `webhook.webhookSecret.name` | string; Name of the secret or configmap which holds the Opsgenie API key | none |
31+
| `webhook.webhookSecret.namespace` | string; Namespace of the secret or configmap | none |
32+
| `webhook.webhookSecret.fieldPath` | string; Key inside the secret or configmap | none |
33+
| `webhook.(*).headers.key` | string; Name for the header key | none |
34+
| `webhook.(*).headers.value` | string; Value of the header | none |
35+
| `webhook.(*).headers.locked` | bool; Is the header value visible in the checklyhq console | none |
36+
| `webhook.(*).queryParameters.key` | string; Name for the query parameter key | none |
37+
| `webhook.(*).queryParameters.value` | string; Value of the query parameter | none |
38+
| `webhook.(*).queryParameters.locked` | bool; Is the query parameter value visible in the checklyhq console | none |
1039

1140
### Email
1241

@@ -18,6 +47,10 @@ kind: AlertChannel
1847
metadata:
1948
name: checkly-operator-test-email
2049
spec:
50+
sendRecovery: false
51+
sendFailure: true
52+
sslExpiry: true
53+
sslExpiryThreshold: 30
2154
email:
2255
address: "[email protected]"
2356
```
@@ -35,6 +68,10 @@ kind: AlertChannel
3568
metadata:
3669
name: checkly-operator-test-opsgenie
3770
spec:
71+
sendRecovery: false
72+
sendFailure: true
73+
sslExpiry: true
74+
sslExpiryThreshold: 30
3875
opsgenie:
3976
apikey:
4077
name: test-secret # Name of the secret or configmap which holds the API key
@@ -44,6 +81,46 @@ spec:
4481
region: "EU" # Your OpsGenie region
4582
```
4683
84+
### Webhook
85+
The webhook integration supports all the fields which are supported by the checkly-go-sdk, see [details](https://pkg.go.dev/github.com/checkly/checkly-go-sdk#AlertChannelWebhook). For other fields and their options, please see [the official docs](https://www.checklyhq.com/docs/alerting-and-retries/webhooks/).
86+
87+
The `WebhookSecret` is an optional field and it requires a kubernetes secret (just like the OpsGenie integration).
88+
89+
Minimum required fields:
90+
* `name` - string
91+
* `url` - string
92+
* `method` - string
93+
94+
```yaml
95+
apiVersion: k8s.checklyhq.com/v1alpha1
96+
kind: AlertChannel
97+
metadata:
98+
name: checkly-operator-test-webhook
99+
spec:
100+
sendRecovery: false
101+
sendFailure: true
102+
sslExpiry: true
103+
sslExpiryThreshold: 30
104+
webhook:
105+
name: foo # Name of the webhook
106+
url: https://foo.bar # URL of the webhook
107+
webhookType : baz # Type of webhook
108+
method: POST # Method of webhook
109+
template: testing # Checkly webhook template
110+
webhookSecret:
111+
name: test-secret # Name of the secret or configmap which holds the webhook secret
112+
namespace: default # Namespace of the secret or configmap
113+
fieldPath: "SECRET_KEY" # Key inside the secret or configmap
114+
headers:
115+
- key: "foo"
116+
value: "bar"
117+
locked: true # Not visible in the UI
118+
queryParameters:
119+
- key: "bar"
120+
value: "baz"
121+
locked: false # Visible in the UI
122+
```
123+
47124
## Referencing
48125

49126
You'll need to reference the name of the alert channel in the group check configuration. See [check-group](check-group.md) for more details.

0 commit comments

Comments
 (0)