Writing an event source using JavaScript¶
This tutorial provides instructions to build an event source in JavaScript and implement it with a ContainerSource or SinkBinding.
- Using a ContainerSource is a simple way to turn any dispatcher container into a Knative event source.
- Using SinkBinding provides a framework for injecting environment variables into any Kubernetes resource that has a
spec.template
and is PodSpecable.
ContainerSource and SinkBinding both work by injecting environment variables to an application. Injected environment variables at minimum contain the URL of a sink that will receive events.
Bootstrapping¶
Create the project and add the dependencies:
npm init -y
npm install cloudevents got --save
Using ContainerSource¶
A ContainerSource creates a container for your event source image and manages this container.
The sink URL where events are posted will be made available to the application through the K_SINK
environment variable by the ContainerSource.
Example¶
The following example event source emits an event to the sink every second:
// File - index.js
const got = require('got');
const { CloudEvent, Emitter, emitterFor } = require('cloudevents');
const K_SINK = process.env['K_SINK'];
K_SINK || logExit('Error: K_SINK Environment variable is not defined');
console.log(`Sink URL is ${K_SINK}`);
const source = 'urn:event:from:heartbeat/example';
const type = 'heartbeat.example';
let eventIndex = 0;
setInterval(() => {
console.log(`Emitting event # ${++eventIndex}`);
// Create a new CloudEvent each second
const event = new CloudEvent({ source, type, data: {'hello': `World # ${eventIndex}`} });
// Emits the 'cloudevent' Node.js event application-wide
event.emit();
}, 1000);
// Create a function that can post an event
const emit = emitterFor(event => {
got.post(K_SINK, event)
.then(response => {
console.log('Event posted successfully');
console.log(response.data);
})
.catch(err => {
console.log('Error during event post');
console.error(err);
});
});
// Send the CloudEvent any time a Node.js 'cloudevent' event is emitted
Emitter.on('cloudevent', emit);
registerGracefulExit();
function registerGracefulExit() {
process.on('exit', logExit);
//catches ctrl+c event
process.on('SIGINT', logExit);
process.on('SIGTERM', logExit);
// catches 'kill pid' (for example: nodemon restart)
process.on('SIGUSR1', logExit);
process.on('SIGUSR2', logExit);
}
function logExit(message = 'Exiting...') {
// Handle graceful exit
console.log(message);
process.exit();
}
# File - Dockerfile
FROM node:16
WORKDIR /usr/src/app
COPY package*.json ./
RUN npm install
COPY . .
EXPOSE 8080
CMD [ "node", "index.js" ]
Procedure¶
Before publishing the ContainerSource, you must build the application image, and push it to a container registry that your cluster can access.
-
Build and push the image:
export REGISTRY=docker.io/myregistry docker build . -t $REGISTRY/node-heartbeat-source:v1 docker push $REGISTRY/node-heartbeat-source:v1
-
Create the event display service which logs any CloudEvents posted to it:
apiVersion: serving.knative.dev/v1 kind: Service metadata: name: event-display spec: template: spec: containers: - image: gcr.io/knative-releases/knative.dev/eventing-contrib/cmd/event_display
-
Create the ContainerSource object:
apiVersion: sources.knative.dev/v1 kind: ContainerSource metadata: name: heartbeat-source spec: template: spec: containers: - image: docker.io/myregistry/node-heartbeat-source:v1 name: heartbeats sink: ref: apiVersion: serving.knative.dev/v1 kind: Service name: event-display
-
Check the logs of the event display service. You will see a new message is pushed every second:
$ kubectl logs -l serving.knative.dev/service=event-display -c user-container ☁️ cloudevents.Event Validation: valid Context Attributes, specversion: 1.0 type: heartbeat.example source: urn:event:from:heartbeat/example id: 47e69d34-def7-449b-8382-3652495f9163 datacontenttype: application/json Data, { "hello": "World 1" }
-
Optional: If you are interested in seeing what is injected into the event source as a
K_SINK
, you can check the logs:$ kubectl logs heartbeat-source-7575c888c7-85w5t Sink URL is http://event-display.default.svc.cluster.local Emitting event #1 Emitting event #2 Event posted successfully Event posted successfully
Using SinkBinding¶
SinkBinding does not create any containers. It injects the sink information to an already existing Kubernetes resources. This is a flexible approach as you can use any Kubernetes PodSpecable object as an event source, such as Deployment, Job, or Knative services.
Procedure¶
-
Create an event display service:
apiVersion: serving.knative.dev/v1 kind: Service metadata: name: event-display spec: template: spec: containers: - image: gcr.io/knative-releases/knative.dev/eventing-contrib/cmd/event_display
-
Create a Kubernetes deployment that runs the event source:
apiVersion: apps/v1 kind: Deployment metadata: name: node-heartbeats-deployment labels: app: node-heartbeats spec: replicas: 2 selector: matchLabels: app: node-heartbeats template: metadata: labels: app: node-heartbeats spec: containers: - name: node-heartbeats image: docker.io/myregistry/node-heartbeat-source:v1 ports: - containerPort: 8080
-
Because the SinkBinding has not yet been created, you will see an error message, because the
K_SINK
environment variable is not yet injected:$ kubectl logs node-heartbeats-deployment-9ffbb644b-llkzk Sink URL is undefined Emitting event #1 Error during event post TypeError [ERR_INVALID_ARG_TYPE]: The "url" argument must be of type string. Received type undefined
-
Create the SinkBinding object:
apiVersion: sources.knative.dev/v1 kind: SinkBinding metadata: name: bind-node-heartbeat spec: subject: apiVersion: apps/v1 kind: Deployment selector: matchLabels: app: node-heartbeats sink: ref: apiVersion: serving.knative.dev/v1 kind: Service name: event-display
You will see the pods are recreated and this time the
K_SINK
environment variable is injected.Also note that since the
replicas
is set to 2, there will be 2 pods that are posting events to the sink.$ kubectl logs event-display-dpplv-deployment-67c9949cf9-bvjvk -c user-container ☁️ cloudevents.Event Validation: valid Context Attributes, specversion: 1.0 type: heartbeat.example source: urn:event:from:heartbeat/example id: 47e69d34-def7-449b-8382-3652495f9163 datacontenttype: application/json Data, { "hello": "World 1" } ☁️ cloudevents.Event Validation: valid Context Attributes, specversion: 1.0 type: heartbeat.example source: urn:event:from:heartbeat/example id: 47e69d34-def7-449b-8382-3652495f9163 datacontenttype: application/json Data, { "hello": "World 1" }