Build and publish Oracle Linux developer container images to GitHub Container Registry #1366
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
name: Build and publish Oracle Linux developer container images to GitHub Container Registry | |
# Builds are triggered either by: | |
# - a push on the main branch with changes in this file. | |
# All container images will be (re)built. | |
# - a push on the main branch with changes the OracleLinuxDevelopers | |
# directory. | |
# Affected container images will be (re)built. | |
# - a manual trigger of the workflow using the API. | |
# Subset of OL version / language can be specified; default is to build | |
# all images. | |
# Images are built for both amd64 and arm64 architectures, except for | |
# - oracledb images (not available on arm) | |
# - php and nodejs on OL7 (packages not available) | |
on: | |
push: | |
branches: | |
- main | |
paths: | |
- 'OracleLinuxDevelopers/**' | |
- '.github/workflows/build-and-push-dev-images.yml' | |
workflow_dispatch: | |
inputs: | |
ol: | |
description: List of ol versions to build | |
default: 'oraclelinux7, oraclelinux8, oraclelinux9' | |
required: false | |
lang: | |
description: List of languages to build | |
default: 'gcc-toolset, golang, nginx, nodejs, php, python, redis, ruby, haproxy, kubectl, helm, ocne-tools' | |
required: false | |
# Default values for the builds triggered by the push event | |
env: | |
ol: 'oraclelinux7, oraclelinux8, oraclelinux9' | |
lang: 'gcc-toolset, golang, nodejs, nginx, php, python, redis, ruby, haproxy, kubectl, helm, ocne-tools' | |
jobs: | |
prepare: | |
name: Create build matrix | |
runs-on: ubuntu-latest | |
outputs: | |
matrix: ${{ steps.build-matrix.outputs.matrix }} | |
skip_build: ${{ steps.build-matrix.outputs.skip_build }} | |
repository_owner: ${{ steps.repository_owner.outputs.repository_owner }} | |
date_stamp: ${{ steps.date_stamp.outputs.date_stamp }} | |
steps: | |
- name: Checkout repo | |
uses: actions/checkout@v4 | |
with: | |
# We need "some" commit history to check for changed files | |
fetch-depth: 32 | |
- name: Build matrix | |
id: build-matrix | |
working-directory: OracleLinuxDevelopers | |
run: | | |
IFS=", " read -r -a ol_list <<< "${{ github.event.inputs.ol || env.ol}}" | |
IFS=", " read -r -a lang_list <<< "${{ github.event.inputs.lang || env.lang}}" | |
changes=$(mktemp) | |
# workflow is only set in the workflow_dispatch event payload | |
workflow="${{ github.event.workflow }}" | |
if [[ -z ${workflow} ]]; then | |
# Push event - retrieve list of changed files | |
git diff --name-only '${{ github.event.before }}..${{ github.event.after }}' > "${changes}" | |
if grep -q build-and-push-dev-images.yml "${changes}"; then | |
echo "PUSH: Action updated, rebuilding all images" | |
build_all=1 | |
else | |
echo "PUSH: Rebuilding changed images only" | |
build_all=0 | |
fi | |
else | |
echo "MANUAL: Rebuilding based on parameters" | |
build_all=1 | |
fi | |
matrix=$( | |
for ol in "${ol_list[@]}"; do | |
pushd "${ol}" >/dev/null || exit 1 | |
for lang in "${lang_list[@]}"; do | |
if [[ -d ${lang} ]]; then | |
pushd "${lang}" >/dev/null || exit 1 | |
for dockerfile in */Dockerfile; do | |
tag=$(dirname "${dockerfile}") | |
if [[ -f ${tag}/.skip-arm64 ]]; then | |
multi=0 | |
arch="linux/amd64" | |
else | |
multi=1 | |
arch="linux/amd64,linux/arm64" | |
fi | |
if [[ ${build_all} -eq 1 ]] || grep -q "${ol}/${lang}/${tag}" "${changes}"; then | |
echo "${ol};${lang};${tag};${arch};${multi}" | |
fi | |
done | |
popd >/dev/null || exit 1 | |
fi | |
done | |
popd >/dev/null || exit 1 | |
done | jq --slurp --raw-input --compact-output ' | |
split("\n") | | |
.[:-1] | | |
map(split(";")) | | |
map({"ol": .[0], "lang": .[1], "tag": .[2], "arch": .[3], "multi": (.[4] == "1")})' | |
) | |
rm "${changes}" | |
if [[ ${matrix} == "[]" ]]; then | |
# Empty array -- change didn't impact any image | |
skip_build=true | |
else | |
skip_build=false | |
matrix=$(jq --compact-output '{ "include": .}' <<<"${matrix}") | |
fi | |
echo "matrix=${matrix}" >> "$GITHUB_OUTPUT" | |
echo "skip_build=${skip_build}" >> "$GITHUB_OUTPUT" | |
- name: Lowercase repository owner | |
id: repository_owner | |
run: | | |
echo "repository_owner=$(echo '${{ github.repository_owner }}' | tr '[:upper:]' '[:lower:]')" >> "$GITHUB_OUTPUT" | |
- name: Date stamp | |
id: date_stamp | |
run: | | |
echo "date_stamp=$(date +'%Y%m%d')" >> "$GITHUB_OUTPUT" | |
build-image: | |
name: Build image | |
needs: [ prepare ] | |
if: always() && needs.prepare.outputs.skip_build == 'false' | |
strategy: | |
matrix: ${{fromJson(needs.prepare.outputs.matrix)}} | |
fail-fast: false | |
runs-on: ubuntu-latest | |
steps: | |
- name: Checkout repo | |
uses: actions/checkout@v4 | |
- name: Set up QEMU | |
uses: docker/setup-qemu-action@v3 | |
with: | |
platforms: arm64 | |
- name: Set up Docker Buildx | |
uses: docker/setup-buildx-action@v3 | |
- name: Log into GitHub Container Registry | |
uses: docker/login-action@v3 | |
with: | |
registry: ghcr.io | |
username: ${{ github.actor }} | |
password: ${{ secrets.GITHUB_TOKEN }} | |
- name: Build image - amd64 | |
uses: docker/build-push-action@v6 | |
with: | |
context: OracleLinuxDevelopers/${{ matrix.ol }}/${{ matrix.lang }}/${{ matrix.tag }} | |
platforms: linux/amd64 | |
push: ${{ github.event_name != 'pull_request' }} | |
tags: | | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-${{ needs.prepare.outputs.date_stamp }}${{ matrix.multi && '-amd64' || '' }}" | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}${{ matrix.multi && '-amd64' || '' }}" | |
- name: Build image - arm64 | |
uses: docker/build-push-action@v6 | |
if: matrix.multi | |
with: | |
context: OracleLinuxDevelopers/${{ matrix.ol }}/${{ matrix.lang }}/${{ matrix.tag }} | |
platforms: linux/arm64 | |
push: ${{ github.event_name != 'pull_request' }} | |
tags: | | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-${{ needs.prepare.outputs.date_stamp }}-arm64" | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-arm64" | |
- name: Manifest | |
if: matrix.multi && github.event_name != 'pull_request' | |
run: | | |
docker buildx imagetools create --tag \ | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-${{ needs.prepare.outputs.date_stamp }}" \ | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-${{ needs.prepare.outputs.date_stamp }}-amd64" \ | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-${{ needs.prepare.outputs.date_stamp }}-arm64" | |
docker buildx imagetools create --tag \ | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}" \ | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-amd64" \ | |
"ghcr.io/${{ needs.prepare.outputs.repository_owner }}/${{ matrix.ol }}-${{ matrix.lang }}:${{ matrix.tag }}-arm64" |