CI/CD Integration
Upload and deploy pipelines as part of your CI/CD flows
GitHub (and other platforms) offer a variety of tools to allow automated execution of arbitrary workflows when new code is committed, or under other conditions such as PR merges. This guide covers how to upload pipelines to Mystic as part of this automation system. We will be covering the use of GitHub actions but the insights are easily transferrable.
All code is available in our example repo here: mystic-ai/ci-cd-example, for this guide.
Overview
Our CI/CD flow works by using the docker buildx action with our python dependencies installed to build the pipeline and then push it to specific tags. In this example we have a src/
directory with out pipeline files that will then be targeted later on.
This guide will not cover the basics on how to use GitHub actions, but if you are unfamiliar with them I recommend reading the GitHub docs here: <https://docs.github.com/en/actions>.
Workflows
We will be considering the simple case of a manually triggered workflow. You can create github actions by placing in yaml configuration files into your repo in the .github/workflows
directory. We will be using two files in our example:
.github/workflows/docker-build.yaml
- This contains the pipeline specific build code that can be used by any github action.github/workflows/manual-docker.yaml
- This is the manual github action that calls the above workflow
Secrets
This example does use the
PIPELINE_API_TOKEN
which you will have to have present in your GitHub actions secrets or Organisation secrets.
Here's the code:
# manual-docker.yaml
name: Manual docker building
on:
workflow_dispatch:
inputs:
pipeline_tag:
description: "Pipeline tag"
required: true
default: "manual"
run-name: "Building pipeline: ${{ github.ref_name }} -> ${{ inputs.pipeline_tag }}"
jobs:
lint:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Check formatting
uses: psf/black@stable
with:
version: 23.12.0
build-pipeline:
needs:
- lint
uses: ./.github/workflows/docker-build.yaml
with:
pipelineTag: ${{ inputs.pipeline_tag }}
secrets:
PIPELINE_API_TOKEN: ${{ secrets.PIPELINE_API_TOKEN }}
# docker-build.yaml
name: Build and push Docker image
on:
workflow_call:
inputs:
pipelineTag:
required: true
type: string
commitRef:
description: "Commit reference to checkout"
type: string
# Empty value uses the default behaviour of actions/[email protected]
default: ""
secrets:
PIPELINE_API_TOKEN:
required: true
description: The API token to authenticate with the pipeline
jobs:
build-push-image:
runs-on: ubuntu-latest
env:
PIPELINE_API_TOKEN: ${{secrets.PIPELINE_API_TOKEN}}
steps:
- uses: actions/checkout@v3
with:
ref: ${{ inputs.commitRef }}
- uses: docker/setup-buildx-action@v2
- name: Setup environment
run: |
pip install -U poetry
poetry install
- name: Build pipeline
run: |
cd src/
poetry run pipeline container build
- name: Push pipeline image
run: |
cd src/
poetry run pipeline container push -p paulh/random-generator:${{ inputs.pipelineTag }} -o
In this example we take a single user input of pipelineTag
in the manual-docker.yaml
file to pass into the pointer definition when we finally push.
How it works
To build and push a pipeline we will need:
- Docker to be installed
poetry
to be running our python virtual environment (you can use another venv but it will be more complicated)pipeline-ai
python SDK installed
Heres the steps in our build/push flow:
steps:
- uses: actions/checkout@v3
with:
ref: ${{ inputs.commitRef }}
- uses: docker/setup-buildx-action@v2
- name: Setup environment
run: |
pip install -U poetry
poetry install
- name: Build pipeline
run: |
cd src/
poetry run pipeline container build
- name: Push pipeline image
run: |
cd src/
poetry run pipeline container push -p paulh/random-generator:${{ inputs.pipelineTag }} -o
This works by:
- Performing a checkout on the target branch to make sure the correct code will be present on the system
- Poetry is installed and then our project dependencies are also installed with
poetry install
- this installspipeline-ai
- We build the container with
poetry run pipeline container build
. By prependingpoetry run
we ensure that we're using the poetry venv. This step also communicates with the local docker installation. Docker is setup in line 5 above when we run- uses: docker/setup-buildx-action@v2
- We push the pipeline to Mystic with a fixed tag that was passed in as an input. The
-o
field is for overwriting existing pointers if they already exist.
One additional useful modification could be to associate a pipeline with a specific commit hash, this is possible by adding an additional pointer to the push command:
...
- name: Build pipeline
run: |
cd src/
poetry run pipeline container build
- name: Push pipeline image
run: |
cd src/
poetry run pipeline container push -p paulh/random-generator:${{ inputs.pipelineTag }} -p paulh/random-generator:${{ inputs.commitRef }} -o
It's possible to do this as the commitRef
is passed as an input from the main manual-docker.yaml
file.
Updated 9 months ago