gitops-promotion is tool to do automatic promotion with a GitOps workflow. It is ideally suited for use with Kubernetes manifests and a controller such as Flux.
gitops-promotion interacts with a Git provider to do automatic propagation of container images across a succession of environments. Supported Git providers:
- GitHub
- Azure Devops
gitops-promotion workflow assumes a separation of one or more "app" repositories, which results in container images, and the repository (or repositories) that hold manifests that describe how those containers are deployed. We refer to this repository as the "GitOps" repository. Assuming a typical dev/qa/production succession of environments, gitops-promotion is meant to support a workflow that looks like this:
- A pipeline in
webui
app repository builds, tests and delivers an image to the container registry. - The new container image triggers a new promotion (
gitops-promotion new
) in the GitOps repository. It creates a new branchpromote/webshop-webui
an auto-merging pull request for the "dev" env. It updates the manifest of the app with the new image. - The auto-merge triggers the promote pipeline (
gitops-promote promote
) in the GitOps repository. This pipeline goes through the same steps as "new" above except that it targets the next environment, in this case the "qa" environment. - The promotion pull request for the "qa" env triggers the "status" pipeline (
gitops-promotions status
). This pipeline checks the status of the "dev" pull request (including any reconciliation status added by Flux) and reports that status as its own. - Assuming the "dev" pull request status is green, the "qa" pull request is merged.
- Steps 4. and 5. are repeated for the "production" environment, but without auto-merge, so that they can be applied at an opportune time.
Conceptually, this means that:
- all new container images are applied to the "dev" environment
- all new container images that are successfully applied will be propagated to the "qa" environment
- pull requests for applying changes to the production environment are automatically created and can be merged by testers or product owners once they have been validated in the "qa" environment.
See the provider-specific sections below for details about how to implement these pipelines.
$ gitops-promotion new --help
Usage of new:
--app string
Name of the application
--group string
Main application group
--provider string
git provider to use (default "azdo")
--tag string
Application version/tag to set
--token string
Access token (PAT) to git provider
The new
command goes through this process:
- creates a new branch
promote/<group>-<app>
(orpromote/<group>-<app>-<env>
ifper-env
is set; resets it if it already exists), - updates the image tag for the app manifest in the first environment listed in the config file to the newly released container image (see below for more info how this works),
- creates an auto-merging pull request,
- Assuming the pull request has no failing checks, it is automatically merged into main, where a service such as Flux can apply it to the first environment.
$ gitops-promotion promote --help
Usage of promote:
--provider string
git provider to use (default "azdo")
--token string
Access token (PAT) to git provider
The promote
command is meant to be used in a pipeline that reacts to merge operations to the main branch that resulted from new
or promote
command. It looks up the pull request and uses the information contained therein to create a new pull request, following the process outlined under the new
command.
$ gitops-promotion status --help
Usage of status:
--provider string
git provider to use (default "azdo")
--token string
Access token (PAT) to git provider
The status
command requests statuses on the merge commit that resulted from the previous' environment's pull request. It looks for a status check with context */<group>-<env>
. This matches the metadata name of a Kustomization resource as reported by the Flux Notification controller (in this case group is "apps"):
If there is no matching status, it then looks on the head commit of "main" branch. If another commit is added to main before Flux has time to consider the merge commit, the merge commit status will never be set, but a relevant status will eventually be set on "main" branch.
The status
command keeps looking for statuses for some time. If there is no status after some minutes, the status
command fails, resulting in a failed check on the pull request, blocking any automatic merging.
$ gitops-promotion feature --help
Usage of feature:
--app string
Name of the application
--group string
Main application group
--provider string
git provider to use (default "azdo")
--tag string
Application version/tag to set
--feature strng
Application feature
--token string
Access token (PAT) to git provider
The feature
command is used to create temporary deployments of applications. It can either overwrite an existing applications image tag, or it can create a new copy of all of the applications manifests. This behavior depends on if featureOverwrite
is enabled or not. Either way a feature will never be promoted.
gitops-promotion assumes a repository with a layout like this (excluding CI pipeline definitions). In Flux, this is referred to as a Monorepo layout:
|-- gitops-promotion.yaml
|-- <group 1>
| |-- <environment 1>
| | |-- ... <-- your Kubernetes YAML here
| | <environment 2>
| <group 2>
| |-- ...
Assuming we are serving a webshop from Kubernetes with three environments, the file structure may looks something like this:
|-- gitops-promotion.yaml
|-- webshop
| |-- dev
| | |-- cart.yaml
| | |-- webui.yaml
| |-- qa
| | |-- cart.yaml
| | |-- webui.yaml
| | production
| | |-- cart.yaml
| | |-- webui.yaml
See below for details about the gitops-promotion.yaml
file.
gitops-promotion uses Flux image-automation-controller to update the Container image tag for containers in your manifests. You annotate the image reference in your manifest (or in your Kustomization image
override) like so, where <group>
and <app>
are arbitrary names that you use to group and name the services gitops-promotion is working with. For more information, see Configure image updates in the Flux documentation.
image: some/image:latest # {"$imagepolicy": "<group>:<app>:tag"}
For example, you may have a very simple manifest like this:
apiVersion: apps/v1
kind: Pod
metadata:
name: webui
spec:
containers:
- name: webui
image: ghcr.io/my-org/webui:1234567 # {"$imagepolicy": "webshop:ui:tag"}
When you have pushed a new image to your container registry, the pipeline runs the following command to start the promotion of your latest image across the configured environments:
gitops-promotion new --provider azdo --token s3cr3t --group webshop --app webui --tag 26f50d84db02
This will instruct gitops-promotion to look up the $imagepolicy
entry webshop:webui:tag
and update the container tag to refer to the tag 26f50d84db02
with the expectation that a GitOps controller such as Flux will react to this change and update the corresponding environment accordingly.
The gitops-promotion.yaml
lists environment names and whether they allow automatic promotion. A typical config file looks like this. gitops-promotion will promote your change across environments in this order.
prflow: per-app
environments:
- name: dev
auto: true
- name: qa
auto: true
- name: prod
auto: false
groups:
apps:
applications:
podinfo:
featureOverwrite: false
featureLabelSelector:
app: podinfo
property | usage |
---|---|
prflow | per-app means later changes will "reset" the single PR for that app, while per-env will upsert a PR that app's PR for a particular environment |
environments[].auto | Whether pull requests for this environment auto-merge or not |
environments[].name | The name for this environment. Must correspond to a directory present in all groups |
Support for Azure DevOps is mature, but alas the documentation is not. TBD.
gitops-promotion has full support for GitHub.
gitops-promotion makes use of PR auto-merge. This requires specific configuration:
- In repository settings, turn on "Allow auto-merge"
- Set up a branch protection rule for your "main" branch:
- "Require a pull request before merging"
- "Require status checks to pass"
- Add the workflow for the
status
commend to "Status checks that are required". In the example below, you would enter "prev-env-status". Bizarrely, the UI will not allow you to enter the name, so you may have to trigger a run once so you can use the search interface.
You can verify that your settings are correct by manually creating a pull request and verify that the button says "Enable auto-merge".
gitops-promotion is available as a Github Action.
Depending on which container registry you are using, you may be able to set up triggers that activates your gitops-promotion workflow. If this is not the case, you can use GitHub repository_dispatch events. These allow GitHub actions on one repository to notify another repository. Use the excellent repository-dispatch GitHub Action for readable YAML. You would add a step at the end of your container-building workflow that looks something like this:
- name: Notify gitops-promotion workflow
uses: peter-evans/repository-dispatch@v1
with:
token: ${{ secrets.GITOPS_REPO_TOKEN }}
repository: my-org/my-gitops
event-type: image-push
client-payload: |
{
"group": "apps",
"app": "my-app",
"tag": "${{ github.sha }}"
}
The repository
parameter holds the repository where you want to run gitops-promotion
. The normal ${{ secrets.GITHUB_TOKEN }}
only has access to the local repository running in which the workflow is running, so we need to set up and pass an access token (GITOPS_REPO_TOKEN) that has access to that repository.
Here is a complete example GitHub workflow for pushing a containerized app to GitHub Container Registry:
on:
push:
branches:
- main
jobs:
build-app:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v1
- name: Login to GitHub Container Registry
uses: docker/login-action@v1
with:
registry: ghcr.io
username: ${{ github.repository_owner }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Build and push
uses: docker/build-push-action@v2
with:
push: true
tags: ghcr.io/${{ github.repository_owner }}/my-app:${{ github.sha }}
- name: Notify gitops-promotion workflow
uses: peter-evans/repository-dispatch@v1
with:
token: ${{ secrets.GITOPS_REPO_TOKEN }}
repository: ${{ github.repository_owner }}/my-gitops
event-type: image-push
client-payload: |
{
"group": "apps",
"app": "my-app",
"tag": "${{ github.sha }}"
}
In your gitops repository, you can react to repository-dispatch
events and trigger promotion:
on:
repository_dispatch:
types:
- image-push
jobs:
new-pr:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
with:
# gitops-promotion currently needs access to history
fetch-depth: 0
- uses: xenitab/[email protected]
with:
token: ${{ secrets.GITHUB_TOKEN }}
action: new
group: ${{ github.event.client_payload.group }}
app: ${{ github.event.client_payload.app }}
tag: ${{ github.event.client_payload.tag }}
This simple example will start the promotion of my-app
onto the first environment defined in the gitops-promotion.yaml
file. In order to promote my-app
to further environments, set up a separate workflow that reacts to merges from previous promotions, like so:
on:
push:
branches:
- main
jobs:
promote-app:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
with:
# gitops-promotion currently needs access to history
fetch-depth: 0
- uses: xenitab/[email protected]
with:
token: ${{ secrets.GITHUB_TOKEN }}
action: promote
In order to block automatic promotion, you can add a status workflow:
on:
pull_request:
branches:
- main
jobs:
prev-env-status:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
with:
# gitops-promotion currently needs access to history
fetch-depth: 0
- uses: xenitab/[email protected]
with:
token: ${{ secrets.GITHUB_TOKEN }}
action: status
For simplicity, the above example uses a Personal Access Token for authentication. However, in a production setup you probably want to use a GitHub App. Once you have set up a GitHub app, you can use the tibdex/github-app-token action to generate a token for the app to access the repository. (In the case of the build-app
job above, you also want to add repository: ${{ github.repository_owner }}/my-gitops
since the token should be valid for the repository we dispatch to.)
# ...
- name: Generate GitHub App token
uses: tibdex/github-app-token@v1
id: generate_token
with:
app_id: ${{ secrets.MY_GITHUB_APP_ID }}
private_key: ${{ secrets.MY_GITHUB_APP_PRIVATE_KEY }}
# It defaults to current repo, so with peter-evans/repository-dispatch you need to specify repo
# repository: ${{ github.repository_owner }}/my-gitops
- uses: xenitab/[email protected]
with:
token: ${{ steps.generate_token.outputs.token }}
# ...
Please note that you will need to make this a required check for merging into main, so it is important that it runs on all pull requests against "main" or your manual pull requests will not be mergeable.
GitHub PR creation says "could not set auto-merge on PR": Your repository is not properly configured to allow pull request auto-merge. Please see the configuration section above for information on how to do this.
You will need pkg-config and libgit2, please install it from your package manager.
The test suite for the GitHub provider requires access to an actual GitHub repository. In order to run these tests, create an empty repository and set up an access key and invoke the tests like so:
env GITHUB_URL='' GITHUB_TOKEN='' go test ./...
The GitHub Action CI runs the tests against https://github.com/gitops-promotion/gitops-promotion-testing.
In order to test interactions manually, you may want to trigger a new promotion. Assuming you are using the example above based on repository-dispatch, the following command will inject a new event:
curl -X POST \
-H "Authorization: token <PAT>" \
-H "Accept: application/vnd.github.v3+json" \
-d '{"event_type": "image-push", "client_payload": {"group": "apps", "app": "my-app", "tag": "123456"}}' \
https://api.github.com/repos/<org>/<repo>/dispatches
In order to emulate a status update from Flux, use the following command:
curl -X POST \
-H "Authorization: token <PAT>" \
-H "Accept: application/vnd.github.v3+json" \
-d '{"state": "success", "context": "kustomization/apps-qa", "description": "reconciliation succeeded"}' \
https://api.github.com/repos/<org>/<repo>/commits/<sha>/statuses