Skip to content

Commit cd46b71

Browse files
committed
Add support for schema_uri validation
Custom schemas can be loaded into the SDK. If a consumed event includes a custom schema, it can be loaded from the SDK local DB and the event can be validated accordingly. Added an example about parsing an event with a custom schema. Signed-off-by: Andrea Frittoli <[email protected]>
1 parent 080d904 commit cd46b71

File tree

65 files changed

+1513
-157
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

65 files changed

+1513
-157
lines changed

.github/workflows/coverage.yml

+1
Original file line numberDiff line numberDiff line change
@@ -39,3 +39,4 @@ jobs:
3939
uses: codecov/codecov-action@e28ff129e5465c2c0dcc6f003fc735cb6ae0c673 # v4.5.0
4040
with:
4141
token: ${{ secrets.CODECOV_TOKEN }}
42+
exclude: docs

docs/README.md

+97-16
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22

33
This folder contains example of how to use this SDK.
44

5+
> **Note** For simplicity, the code below does not include error handling. The go files in example folders includes it.
6+
57
## Create a Custom CDEvent
68

79
If a tool wants to emit events that are not supported by the CDEvents specification,
@@ -17,21 +19,24 @@ happens, but CDEvents does not define any quota related subject.
1719

1820
```golang
1921
type Quota struct {
20-
User string `json:"user,omitempty"` // The use the quota applies ot
21-
Limit string `json:"limit,omitempty"` // The limit enforced by the quota e.g. 100Gb
22-
Current int `json:"current,omitempty"` // The current % of the quota used e.g. 90%
23-
Threshold int `json:"threshold,omitempty"` // The threshold for warning event e.g. 85%
24-
Level string `json:"level,omitempty"` // INFO: <threshold, WARNING: >threshold, <quota, CRITICAL: >quota
22+
User string `json:"user,omitempty"` // The use the quota applies ot
23+
Limit string `json:"limit,omitempty"` // The limit enforced by the quota e.g. 100Gb
24+
Current int `json:"current,omitempty"` // The current % of the quota used e.g. 90%
25+
Threshold int `json:"threshold,omitempty"` // The threshold for warning event e.g. 85%
26+
Level string `json:"level,omitempty"` // INFO: <threshold, WARNING: >threshold, <quota, CRITICAL: >quota
2527
}
2628
```
29+
2730
For this scenario we will need a few imports:
2831

2932
```golang
3033
import (
3134
"context"
32-
"fmt"
35+
"fmt"
3336
"log"
37+
"os"
3438

39+
examples "github.com/cdevents/sdk-go/docs/examples"
3540
cdevents "github.com/cdevents/sdk-go/pkg/api"
3641
cdeventsv04 "github.com/cdevents/sdk-go/pkg/api/v04"
3742
cloudevents "github.com/cloudevents/sdk-go/v2"
@@ -63,9 +68,7 @@ quotaRule123 := Quota{
6368

6469
// Create the base event
6570
event, err := cdeventsv04.NewCustomTypeEvent()
66-
if err != nil {
67-
log.Fatalf("could not create a cdevent, %v", err)
68-
}
71+
examples.PanicOnError(err, "could not create a cdevent")
6972
event.SetEventType(eventType)
7073

7174
// Set the required context fields
@@ -79,17 +82,23 @@ event.SetSubjectContent(quotaRule123)
7982
// to the event so that the receiver may validate custom fields like
8083
// the event type and subject content
8184
event.SetSchemaUri("https://myregistry.dev/schemas/cdevents/quota-exceeded/0_1_0")
85+
86+
// The event schema needs to be loaded, so the SDK may validate
87+
// In this example, the schema is located in the same folder as
88+
// the go code
89+
customSchema, err := os.ReadFile("myregistry-quotaexceeded_schema.json")
90+
examples.PanicOnError(err, "cannot load schema file")
91+
92+
err = cdevents.LoadJsonSchema(customSchemaUri, customSchema)
93+
examples.PanicOnError(err, "cannot load the custom schema file")
8294
```
8395

8496
To see the event, let's render it as JSON and log it:
8597

8698
```golang
8799
// Render the event as JSON
88100
eventJson, err := cdevents.AsJsonString(event)
89-
if err != nil {
90-
log.Fatalf("failed to marshal the CDEvent, %v", err)
91-
}
92-
// Print the event
101+
examples.PanicOnError(err, "failed to marshal the CDEvent")
93102
fmt.Printf("%s", eventJson)
94103
```
95104

@@ -123,10 +132,11 @@ if result := c.Send(ctx, *ce); cloudevents.IsUndelivered(result) {
123132
}
124133
```
125134

126-
The whole code of is available under [`examples/custom.go`](./examples/custom.go):
135+
The whole code of is available under [`examples/custom`](./examples/custom/main.go):
127136

128137
```shell
129-
➜ go run custom.go | jq .
138+
cd examples/custom
139+
➜ go run main.go | jq .
130140
{
131141
"context": {
132142
"version": "0.4.1",
@@ -149,4 +159,75 @@ The whole code of is available under [`examples/custom.go`](./examples/custom.go
149159
}
150160
}
151161
}
152-
```
162+
```
163+
164+
## Consume a CDEvent with a Custom Schema
165+
166+
CDEvents producers may include a `schemaUri` in their events. The extra schema **must** comply with the CDEvents schema and may add additional rules on top.
167+
The `schemaUri` field includes the `$id` field of the custom schema and can be used for different purposes:
168+
* specify the format of the data included in the `customData` field
169+
* specify the format of the subject content of custom events
170+
* refine the format of one or more fields of a specific CDEvent
171+
172+
In this examples, the custom schema is used to define the format of the `customData` for a `change.created` events, which corresponds to the following golang `struct`:
173+
174+
```golang
175+
type ChangeData struct {
176+
User string `json:"user"` // The user that created the PR
177+
Assignee string `json:"assignee,omitempty"` // The user assigned to the PR (optional)
178+
Head string `json:"head"` // The head commit (sha) of the PR
179+
Base string `json:"base"` // The base commit (sha) for the PR
180+
}
181+
```
182+
183+
The goal of this example is to consume (parse) an event with a custom schema and validate it. In the example we load the event from disk. In real life the event will be typically received over the network or extracted from a database.
184+
185+
For this scenario we will need a few imports:
186+
187+
```golang
188+
import (
189+
"context"
190+
"encoding/json"
191+
"fmt"
192+
"log"
193+
"os"
194+
195+
examples "github.com/cdevents/sdk-go/docs/examples"
196+
cdevents "github.com/cdevents/sdk-go/pkg/api"
197+
cdevents04 "github.com/cdevents/sdk-go/pkg/api/v04"
198+
cloudevents "github.com/cloudevents/sdk-go/v2"
199+
)
200+
```
201+
202+
Before parsing an event with a custom schema, it's required to load the schema into the SDK. This avoids having to download and compile the schema every time a message is parsed.
203+
204+
```golang
205+
// Load and register the custom schema
206+
customSchema, err := os.ReadFile("changecreated_schema.json")
207+
208+
// Unmarshal the schema to extract the $id. The $id can also be hardcoded as a const
209+
eventAux := &struct {
210+
Id string `json:"$id"`
211+
}{}
212+
err = json.Unmarshal(customSchema, eventAux)
213+
err = cdevents.LoadJsonSchema(eventAux.Id, customSchema)
214+
```
215+
216+
Once the schema is loaded, it's possible to parse the event itself.
217+
In this case we know that the event is in the v0.4 version format, so we use the corresponding API.
218+
219+
```golang
220+
// Load, unmarshal and validate the event
221+
eventBytes, err := os.ReadFile("changecreated.json")
222+
event, err := cdevents04.NewFromJsonBytes(eventBytes)
223+
224+
err = cdevent.Validate(event)
225+
if err != nil {
226+
log.Fatalf("cannot validate event %v: %v", event, err)
227+
}
228+
229+
// Print the event
230+
eventJson, err := cdevents.AsJsonString(event)
231+
examples.PanicOnError(err, "failed to marshal the CDEvent")
232+
fmt.Printf("%s\n\n", eventJson)
233+
```

docs/examples/custom.go

-105
This file was deleted.

docs/examples/custom/main.go

+88
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
package main
2+
3+
import (
4+
"context"
5+
"fmt"
6+
"log"
7+
"os"
8+
9+
examples "github.com/cdevents/sdk-go/docs/examples"
10+
cdevents "github.com/cdevents/sdk-go/pkg/api"
11+
cdeventsv04 "github.com/cdevents/sdk-go/pkg/api/v04"
12+
cloudevents "github.com/cloudevents/sdk-go/v2"
13+
)
14+
15+
const customSchemaUri = "https://myregistry.dev/schemas/cdevents/quota-exceeded/0_1_0"
16+
17+
type Quota struct {
18+
User string `json:"user"` // The use the quota applies ot
19+
Limit string `json:"limit"` // The limit enforced by the quota e.g. 100Gb
20+
Current int `json:"current"` // The current % of the quota used e.g. 90%
21+
Threshold int `json:"threshold"` // The threshold for warning event e.g. 85%
22+
Level string `json:"level"` // INFO: <threshold, WARNING: >threshold, <quota, CRITICAL: >quota
23+
}
24+
25+
func main() {
26+
var ce *cloudevents.Event
27+
var c cloudevents.Client
28+
29+
// Define the event type
30+
eventType := cdevents.CDEventType{
31+
Subject: "quota",
32+
Predicate: "exceeded",
33+
Version: "0.1.0",
34+
Custom: "myregistry",
35+
}
36+
37+
// Define the content
38+
quotaRule123 := Quota{
39+
User: "heavy_user",
40+
Limit: "50Tb",
41+
Current: 90,
42+
Threshold: 85,
43+
Level: "WARNING",
44+
}
45+
46+
// Create the base event
47+
event, err := cdeventsv04.NewCustomTypeEvent()
48+
examples.PanicOnError(err, "could not create a cdevent")
49+
event.SetEventType(eventType)
50+
51+
// Set the required context fields
52+
event.SetSubjectId("quotaRule123")
53+
event.SetSource("my/first/cdevent/program")
54+
55+
// Set the required subject fields
56+
event.SetSubjectContent(quotaRule123)
57+
event.SetSchemaUri(customSchemaUri)
58+
59+
// Print the event
60+
eventJson, err := cdevents.AsJsonString(event)
61+
examples.PanicOnError(err, "failed to marshal the CDEvent")
62+
fmt.Printf("%s", eventJson)
63+
64+
// To validate the event, we need to load its custom schema
65+
customSchema, err := os.ReadFile("myregistry-quotaexceeded_schema.json")
66+
examples.PanicOnError(err, "cannot load schema file")
67+
68+
err = cdevents.LoadJsonSchema(customSchemaUri, customSchema)
69+
examples.PanicOnError(err, "cannot load the custom schema file")
70+
71+
ce, err = cdevents.AsCloudEvent(event)
72+
examples.PanicOnError(err, "failed to create cloudevent")
73+
74+
// Set send options
75+
source, err := examples.CreateSmeeChannel()
76+
examples.PanicOnError(err, "failed to create a smee channel")
77+
ctx := cloudevents.ContextWithTarget(context.Background(), *source)
78+
ctx = cloudevents.WithEncodingBinary(ctx)
79+
80+
c, err = cloudevents.NewClientHTTP()
81+
examples.PanicOnError(err, "failed to create the CloudEvents client")
82+
83+
// Send the CloudEvent
84+
// c is a CloudEvent client
85+
if result := c.Send(ctx, *ce); cloudevents.IsUndelivered(result) {
86+
log.Fatalf("failed to send, %v", result)
87+
}
88+
}

0 commit comments

Comments
 (0)