This Keptn service introduces a radical new approach to running tasks with keptn. It provides the means to run any container as a Kubernetes Job orchestrated by keptn.
The job-executor-service aims to tackle several current pain points with the current approach of services running in the keptn ecosystem:
Problem | Solution |
---|---|
Keptn services are constantly running while listening for cloud events coming in over NATS. This consumes unnecessary resources on the Kubernetes cluster. | By running the defined keptn tasks as short-lived workloads (Kubernetes Jobs), they just consume resources while the task is executed. |
Whenever some new functionality should be triggered by keptn, a new keptn service needs to be written. It usually wraps the functionality of the wanted framework and executes it under the hood. The downside: The code of the new keptn service needs to be written and maintained. This effort scales linearly with the amount of services. | This service can execute any framework with just a few lines of yaml configuration. No need to write or maintain any new code. |
Keptn services usually filter for a static list of events the trigger the included functionality. This is not configurable. Whenever the service should listen to a new event, the code of the service needs to be changed. | The Job Executor Service provides the means to trigger a task execution for any keptn event. This is done by matching a jsonpath to the received event payload. |
Keptn services are usually opinionized on how your framework execution looks like. E.g. the locust service just executes three different (statically named) files depending on the test strategy in the shipyard. It is not possible to write tests consisting of multiply files. | This service provides the possibility to write any specified file from the keptn git repository into a mounted folder (/keptn ) of the Kubernetes job. This is done by a initcontainer running before the specified image. |
Support for new functionality in keptn needs to be added to each keptn service individually. E.g. the new secret functionality needs to be included into all of the services running in the keptn execution plane. This slows down the availability of this new feature. | The Job Executor Service is a single service which provides the means to run any workload orchestrated by keptn. So, it is possible to support new functionality of keptn just once in this service - and all workloads profit from it. E.g. in the case of the secret functionality, one just needs to support it in this service and suddenly all the triggered Kubernetes Jobs have the correct secrets attached to it as environment variables. |
Just put a file into the keptn git repository (in folder <service>/job/config.yaml
) to specify
- the containers which should be run as Kubernetes Jobs and
- the events for which they should be triggered.
apiVersion: v1
actions:
- name: "Run locust"
events:
- name: "sh.keptn.event.test.triggered"
jsonpath:
property: "$.data.test.teststrategy"
match: "health"
- name: "sh.keptn.event.test.triggered"
jsonpath:
property: "$.data.test.teststrategy"
match: "load"
tasks:
- name: "Run locust smoke tests"
files:
- locust/basic.py
- locust/import.py
- locust/locust.conf
image: "locustio/locust"
cmd: "locust --config /keptn/locust/locust.conf -f /keptn/locust/basic.py --host $(HOST)"
env:
- name: HOST
value: "$.data.deployment.deploymentURIsLocal[0]"
valueFrom: event
The tasks of an action are executed if the event name matches. Wildcards can also be used, e.g.
- name: "sh.keptn.event.*.triggered"
Would match events sh.keptn.event.test.triggered
, sh.keptn.event.deployment.triggered
and so on.
Optionally the following section can be added to an event:
jsonpath:
property: "$.data.test.teststrategy"
match: "locust"
If the service receives an event which matches the name, and the jsonpath match expression, the specified tasks are executed. E.g. the following cloud event would match the jsonpath above:
{
"type": "sh.keptn.event.test.triggered",
"specversion": "1.0",
"source": "test-events",
"id": "f2b878d3-03c0-4e8f-bc3f-454bc1b3d79b",
"time": "2019-06-07T07:02:15.64489Z",
"contenttype": "application/json",
"shkeptncontext": "08735340-6f9e-4b32-97ff-3b6c292bc50i",
"data": {
"project": "sockshop",
"stage": "dev",
"service": "carts",
"labels": {
"testId": "4711",
"buildId": "build-17",
"owner": "JohnDoe"
},
"status": "succeeded",
"result": "pass",
"test": {
"teststrategy": "locust"
}
}
}
The configuration contains the following section:
tasks:
- name: "Run locust smoke tests"
files:
- locust/basic.py
- locust/import.py
- locust/locust.conf
image: "locustio/locust"
cmd: "locust --config /keptn/locust/locust.conf -f /keptn/locust/basic.py --host $(HOST)"
It contains the tasks which should be executed as Kubernetes job. The service schedules a different job for each of
these tasks in the order, they are listed within the config. The service waits for the successful execution of all the
tasks to respond with a StatusSucceeded
finished event. When one of the events fail, it responds with StatusErrored
finished cloud event.
In the env
section of a task, a list of environment variables can be declared, with their source either from the
incoming cloud event (valueFrom: event
) or from kubernetes secrets (valueFrom: secret
).
Environment variables in cmd
can be accessed with parentheses, e.g. "$(HOST)"
. This is required for the variable to
be expanded in the command.
The following environment variable has the name HOST
, and the value is whatever the given
jsonpath $.data.deployment.deploymentURIsLocal[0]
resolves to.
cmd: "locust --config /keptn/locust/locust.conf -f /keptn/locust/basic.py --host $(HOST)"
env:
- name: HOST
value: "$.data.deployment.deploymentURIsLocal[0]"
valueFrom: event
In the above example the json path for HOST
would resolve into https://keptn.sh
for the below event
{
"data": {
"deployment": {
"deploymentNames": [
"user_managed"
],
"deploymentURIsLocal": [
"https://keptn.sh"
],
"deploymentURIsPublic": [
""
],
"deploymentstrategy": "user_managed",
"gitCommit": "eb5fc3d5253b1845d3d399c880c329374bbbb30e"
},
"message": "",
"project": "sockshop",
"stage": "dev",
"service": "carts",
"status": "succeeded",
"test": {
"teststrategy": "health"
}
},
"id": "4fe1eed1-49e2-49a9-91af-a42c8b0f7811",
"source": "shipyard-controller",
"specversion": "1.0",
"time": "2021-05-13T07:46:09.546Z",
"type": "sh.keptn.event.test.triggered",
"shkeptncontext": "138f7bf1-f027-42c4-b705-9033b5f5871e"
}
The following configuration looks up a kubernetes secret with the name locust-secret
and all key/value pairs of the
secret will be available as separate environment variables in the job.
cmd: "locust --config /keptn/locust/locust.conf -f /keptn/locust/$(FILE) --host $(HOST)"
env:
- name: locust-secret
valueFrom: secret
With the secret below, there will be two environment variables available in the job. HOST
with the
value https://keptn.sh
and FILE
with the value basic.py
kubectl -n keptn create secret generic locust-secret --from-literal=HOST=https://keptn.sh --from-literal=FILE=basic.py -oyaml --dry-run=client
apiVersion: v1
data:
FILE: YmFzaWMucHk=
HOST: aHR0cHM6Ly9rZXB0bi5zaA==
kind: Secret
metadata:
creationTimestamp: null
name: locust-secret
namespace: keptn
Files can be added to your running tasks by specifying them in the files
section of your tasks:
files:
- locust/basic.py
- locust/import.py
- locust/locust.conf
This is done by using an initcontainer
for the scheduled Kubernetes Job which prepares the èmptyDir
volume mounted
to the Kubernetes Job. Within the Job itself, the files will be available within the keptn
folder. The naming of the
files and the location will be preserved.
When using these files in your container command, please make sure to reference them by prepending the keptn
path.
E.g.:
cmd: "locust --config /keptn/locust/locust.conf -f /keptn/locust/basic.py"
Actions can be run in silent mode, meaning no .started/.finished
events will be sent by the job-executor-service. This
is particular useful when not matching .triggered
events but e.g. .finished
events where responding
with .started/.finished
events does not make sense. To enable silent mode simply set it to true for the corresponding
action. By default silent mode is disabled for each action.
actions:
- name: "Run locust"
silent: true
If you are using the service in a remote control plane setup make sure the distributor is configured to forward all
events used in the job/config.yaml
. Just edit the PUBSUB_TOPIC
environment variable in the distributor deployment
configuration to fit your needs.
- Run the helm service as a kubernetes job and limit the permissions of it by assigning a different service account to it.
- Execute infrastructure as code with keptn (e.g. run terraform, pulumi, etc)
- Run kaniko inside the cluster and build and push your images. Suddenly, keptn turns into your CI
- ...
The credits of this service heavily go to @thschue and @yeahservice who originally came up with this idea. 🚀
Keptn Version | Job-Executor-Service Docker Image |
---|---|
0.8.3 | keptnsandbox/job-executor-service:0.1.0 |
0.8.3 | keptnsandbox/job-executor-service:0.1.1 |
The job-executor-service can be installed as a part of Keptn's uniform.
To deploy the current version of the job-executor-service in your Keptn Kubernetes cluster, apply
the deploy/service.yaml
file:
kubectl apply -f deploy/service.yaml
This should install the job-executor-service
together with a Keptn distributor
into the keptn
namespace, which you
can verify using
kubectl -n keptn get deployment job-executor-service -o wide
kubectl -n keptn get pods -l run=job-executor-service
Adapt and use the following command in case you want to up- or downgrade your installed version (specified by
the $VERSION
placeholder):
kubectl -n keptn set image deployment/job-executor-service job-executor-service=keptnsandbox/job-executor-service:$VERSION --record
To delete a deployed job-executor-service, use the file deploy/*.yaml
files from this repository and delete the
Kubernetes resources:
kubectl delete -f deploy/service.yaml
Development can be conducted using any GoLang compatible IDE/editor (e.g., Jetbrains GoLand, VSCode with Go plugins).
It is recommended to make use of branches as follows:
master
contains the latest potentially unstable versionrelease-*
contains a stable version of the service (e.g.,release-0.1.0
contains version 0.1.0)- create a new branch for any changes that you are working on, e.g.,
feature/my-cool-stuff
orbug/overflow
- once ready, create a pull request from that branch back to the
master
branch
When writing code, it is recommended to follow the coding style suggested by the Golang community.
- Build the binary:
go build -ldflags '-linkmode=external' -v -o job-executor-service
- Run tests:
go test -race -v ./...
- Build the docker image:
docker build . -t keptnsandbox/job-executor-service:dev
(Note: Ensure that you use the correct DockerHub account/organization) - Run the docker image locally:
docker run --rm -it -p 8080:8080 keptnsandbox/job-executor-service:dev
- Push the docker image to DockerHub:
docker push keptnsandbox/job-executor-service:dev
(Note: Ensure that you use the correct DockerHub account/organization) - Deploy the service using
kubectl
:kubectl apply -f deploy/
- Delete/undeploy the service using
kubectl
:kubectl delete -f deploy/
- Watch the deployment using
kubectl
:kubectl -n keptn get deployment job-executor-service -o wide
- Get logs using
kubectl
:kubectl -n keptn logs deployment/job-executor-service -f
- Watch the deployed pods using
kubectl
:kubectl -n keptn get pods -l run=job-executor-service
- Deploy the service
using Skaffold:
skaffold run --default-repo=your-docker-registry --tail
(Note: Replaceyour-docker-registry
with your DockerHub username; also make sure to adapt the image name in skaffold.yaml)
We have dummy cloud-events in the form of RFC 2616 requests in the test-events/ directory. These can be easily executed using third party plugins such as the Huachao Mao REST Client in VS Code.
This repo uses reviewdog for automated reviews of Pull Requests.
You can find the details in .github/workflows/reviewdog.yml.
This repo has automated unit tests for pull requests.
You can find the details in .github/workflows/tests.yml.
This repo uses GH Actions and Workflows to test the code and automatically build docker images.
Docker Images are automatically pushed based on the configuration done in .ci_env and the two GitHub Secrets
REGISTRY_USER
- your DockerHub usernameREGISTRY_PASSWORD
- a DockerHub access token (alternatively, your DockerHub password)
It is assumed that the current development takes place in the master branch (either via Pull Requests or directly).
To make use of the built-in automation using GH Actions for releasing a new version of this service, you should
- branch away from master to a branch called
release-x.y.z
(wherex.y.z
is your version), - write release notes in the releasenotes/ folder,
- update the compatibility matrix,
- check the output of GH Actions builds for the release branch,
- verify that your image was built and pushed to DockerHub with the right tags,
- update the image tags for
job-executor-service
andjob-executor-service-initcontainer
in deploy/service.yaml, helm/Chart.yaml, helm/values.yaml and helm/templates/configmap.yaml, - test your service against a working Keptn installation.
If any problems occur, fix them in the release branch and test them again.
Once you have confirmed that everything works and your version is ready to go, you should
- create a new release on the release branch using the GitHub releases page, and
- merge any changes from the release branch back to the master branch.
Please find more information in the LICENSE file.