From 2a3e5690cbb8a667ab26acd1e09a6772d9401ab6 Mon Sep 17 00:00:00 2001 From: Jordan Dubrick Date: Thu, 6 Jun 2024 15:19:32 -0400 Subject: [PATCH] Enable Multi-Arch Building and Pushing for registry viewer (#125) * update projects to use multi arch build Signed-off-by: Jordan Dubrick * update dockerfile to customize timeout value Signed-off-by: Jordan Dubrick * add new script for building/pushing multi arch registry viewer Signed-off-by: Jordan Dubrick * free disk space and add qemu emulator Signed-off-by: Jordan Dubrick * use new build/push script and add qemu emulator Signed-off-by: Jordan Dubrick * add buildx setup to actions Signed-off-by: Jordan Dubrick * pin sha for buildx setup Signed-off-by: Jordan Dubrick * increase parallel to 4 Signed-off-by: Jordan Dubrick * break nx command into 2 separate commands Signed-off-by: Jordan Dubrick --------- Signed-off-by: Jordan Dubrick --- .github/workflows/ci.yaml | 26 +++++++++++- .github/workflows/pushimage-next.yml | 10 +++-- Dockerfile | 8 +++- apps/landing-page/project.json | 12 +++++- apps/registry-viewer/project.json | 12 +++++- scripts/build_multi_arch.sh | 59 ++++++++++++++++++++++++++++ 6 files changed, 116 insertions(+), 11 deletions(-) create mode 100644 scripts/build_multi_arch.sh diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index df398e94..97144189 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -99,6 +99,23 @@ jobs: uses: actions/checkout@c85c95e3d7251135ab7dc9ce3241c5835cc595a9 # v3.5.3 with: fetch-depth: 0 + + - name: Free Disk Space + uses: jlumbroso/free-disk-space@54081f138730dfa15788a46383842cd2f914a1be #v1.3.1 + with: + tool-cache: false + android: true + dotnet: true + haskell: true + large-packages: true + docker-images: true + swap-storage: true + + - name: Set up QEMU # Enables arm64 image building + uses: docker/setup-qemu-action@68827325e0b33c7199eb31dd4e31fbe9023e06e3 #v3.0.0 + + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@d70bba72b1f3fd22344832f00baa16ece964efeb #v3.3.0 - uses: actions/setup-node@8f152de45cc393bb48ce5d89d36b731f54556e65 #v4.0.0 with: @@ -111,9 +128,14 @@ jobs: run: | yarn install --frozen-lockfile - - name: Docker Build + - name: Docker Build for linux/amd64 run: | - yarn nx affected --target=docker-build --parallel=3 + yarn nx affected --target=docker-build-amd64 --parallel=3 + + - name: Docker Build for linux/arm64 + run: | + yarn nx affected --target=docker-build-arm64 --parallel=3 + e2e: name: E2E diff --git a/.github/workflows/pushimage-next.yml b/.github/workflows/pushimage-next.yml index d4c59bfc..90ecebb7 100644 --- a/.github/workflows/pushimage-next.yml +++ b/.github/workflows/pushimage-next.yml @@ -31,16 +31,18 @@ jobs: steps: - name: Check out devfile web source code uses: actions/checkout@c85c95e3d7251135ab7dc9ce3241c5835cc595a9 # v3.5.3 + - name: Set up QEMU # Enables arm64 image building + uses: docker/setup-qemu-action@68827325e0b33c7199eb31dd4e31fbe9023e06e3 #v3.0.0 + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@d70bba72b1f3fd22344832f00baa16ece964efeb #v3.3.0 - name: Login to Quay uses: docker/login-action@465a07811f14bebb1938fbed4728c6a1ff8901fc # v2.2.0 with: registry: quay.io username: ${{ secrets.QUAY_USERNAME }} password: ${{ secrets.QUAY_PASSWORD }} - - name: Build the registry viewer image - run: bash ./scripts/build_viewer.sh - - name: Push the registry viewer image - run: bash ./scripts/push.sh registry-viewer:latest quay.io/devfile/registry-viewer:next + - name: Build and push the registry viewer image + run: bash ./scripts/build_multi_arch.sh dispatch: needs: registry-viewer-build strategy: diff --git a/Dockerfile b/Dockerfile index 8223272a..66009a25 100644 --- a/Dockerfile +++ b/Dockerfile @@ -33,6 +33,12 @@ ARG NEXT_PUBLIC_DOCSEARCH_APP_ID ARG NEXT_PUBLIC_DOCSEARCH_API_KEY ARG NEXT_PUBLIC_DOCSEARCH_INDEX_NAME +# Building different architectures via emulation is slow with Yarn +# This increases the timeout period so it can properly download dependencies +# Value is in milliseconds and is set to 60 minutes +# To increase/decrease you can override via --build-arg YARN_TIMEOUT=x in your build command +ARG YARN_TIMEOUT=3600000 + # Check if the PROJECT_NAME build argument is set RUN \ if [ "$PROJECT_NAME" == "landing-page" ] || [ "$PROJECT_NAME" == "registry-viewer" ]; then echo "Building project \"${PROJECT_NAME}\"."; \ @@ -43,7 +49,7 @@ WORKDIR /app # Install dependencies COPY package.json yarn.lock* ./ RUN \ - if [ -f yarn.lock ]; then yarn --frozen-lockfile; \ + if [ -f yarn.lock ]; then yarn --frozen-lockfile --network-timeout $YARN_TIMEOUT; \ else echo "Lockfile not found." && exit 1; \ fi diff --git a/apps/landing-page/project.json b/apps/landing-page/project.json index bbe95a5a..a3f64bc2 100644 --- a/apps/landing-page/project.json +++ b/apps/landing-page/project.json @@ -59,10 +59,18 @@ "command": "cd ./apps/landing-page/ && next-sitemap --config ./next-sitemap.config.mjs && cp -a ./dist/public/. ./dist/exported" } }, - "docker-build": { + "docker-build-amd64": { "executor": "nx:run-commands", + "description": "Build docker image for the linux/amd64 architecture", "options": { - "command": "docker build -t nextjs-docker . --build-arg PROJECT_NAME=landing-page" + "command": "docker build --platform=linux/amd64 -t nextjs-docker-amd64 . --build-arg PROJECT_NAME=landing-page" + } + }, + "docker-build-arm64": { + "executor": "nx:run-commands", + "description": "Build docker image for the linux/arm64 architecture", + "options": { + "command": "docker build --platform=linux/arm64 -t nextjs-docker-arm64 . --build-arg PROJECT_NAME=landing-page" } }, "test": { diff --git a/apps/registry-viewer/project.json b/apps/registry-viewer/project.json index 09ec22d2..a08de6c4 100644 --- a/apps/registry-viewer/project.json +++ b/apps/registry-viewer/project.json @@ -58,10 +58,18 @@ "command": "cd ./apps/registry-viewer/ && next-sitemap --config ./next-sitemap.config.mjs && cp -a ./dist/public/. ./dist/exported" } }, - "docker-build": { + "docker-build-amd64": { "executor": "nx:run-commands", + "description": "Build docker image for the linux/amd64 architecture", "options": { - "command": "docker build -t nextjs-docker . --build-arg PROJECT_NAME=registry-viewer" + "command": "docker build --platform=linux/amd64 -t nextjs-docker-amd64 . --build-arg PROJECT_NAME=registry-viewer" + } + }, + "docker-build-arm64": { + "executor": "nx:run-commands", + "description": "Build docker image for the linux/arm64 architecture", + "options": { + "command": "docker build --platform=linux/arm64 -t nextjs-docker-arm64 . --build-arg PROJECT_NAME=registry-viewer" } }, "test": { diff --git a/scripts/build_multi_arch.sh b/scripts/build_multi_arch.sh new file mode 100644 index 00000000..053e58e5 --- /dev/null +++ b/scripts/build_multi_arch.sh @@ -0,0 +1,59 @@ +#!/bin/sh + +# +# Copyright Red Hat +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +ABSOLUTE_PATH="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" +BUILD_DIR=$ABSOLUTE_PATH/.. +# Due to command differences between podman and docker we need to separate the process +# for creating and adding images to a multi-arch manifest +podman=${USE_PODMAN:-false} +# Base Repository +BASE_REPO="quay.io/devfile/registry-viewer" +BASE_TAG="next" +DEFAULT_IMG="$BASE_REPO:$BASE_TAG" +# Platforms to build for +PLATFORMS="linux/amd64,linux/arm64" + +if [ ${podman} == true ]; then + echo "Executing with podman" + + podman manifest create "$DEFAULT_IMG" + + podman build --platform="$PLATFORMS" --manifest "$DEFAULT_IMG" "$BUILD_DIR" \ + --no-cache \ + --build-arg PROJECT_NAME=registry-viewer \ + --build-arg NEXT_PUBLIC_BASE_PATH=${NEXT_PUBLIC_BASE_PATH:-"/viewer"} + + podman manifest push "$DEFAULT_IMG" + + podman manifest rm "$DEFAULT_IMG" + +else + echo "Executing with docker" + + docker buildx create --name registry-viewer-builder + + docker buildx use registry-viewer-builder + + docker buildx build --push --platform="$PLATFORMS" --tag "$DEFAULT_IMG" "$BUILD_DIR" \ + --no-cache \ + --provenance=false \ + --build-arg PROJECT_NAME=registry-viewer \ + --build-arg NEXT_PUBLIC_BASE_PATH=${NEXT_PUBLIC_BASE_PATH:-"/viewer"} + + docker buildx rm registry-viewer-builder + +fi \ No newline at end of file