Skip to content

Commit

Permalink
[github-actions] build docker image then run arm-gcc with it
Browse files Browse the repository at this point in the history
This changes the relationship of the workflows so that the `arm-gcc` job
in `build.yml` is run using the docker image generated by the docker
`build-image` job.

Previously, the `arm-gcc` job would just use the `:latest` tag of
`siliconlabsinc/ot-efr32-dev` from DockerHub. This presents a problem
when there are changes to the bootstrap script which are required for
the build script to succeed. These changes would not be present in the
currently published `siliconlabsinc/ot-efr32-dev:latest` since that
image was build off of `SiliconLabs/ot-efr32:main`, not the current branch.
  • Loading branch information
lmnotran committed Jun 21, 2024
1 parent 7616b37 commit 8d94119
Show file tree
Hide file tree
Showing 7 changed files with 239 additions and 116 deletions.
111 changes: 64 additions & 47 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -40,13 +40,22 @@ concurrency:
group: ${{ github.workflow }}-${{ github.event.pull_request.number || (github.repository == 'openthread/ot-efr32' && github.run_id) || github.ref }}
cancel-in-progress: true

env:
TOOLCHAIN_DIR: .toolchain
DOCKER_IMAGE_SHA_TAG: siliconlabsinc/ot-efr32-dev:${{ github.sha }}

jobs:
docker:
name: Docker
uses: ./.github/workflows/docker.yml

arm-gcc:
name: arm-gcc-${{ matrix.gcc_ver }}
name: Arm GNU Toolchain ${{ matrix.gcc_ver }}
runs-on: ubuntu-22.04
container:
image: siliconlabsinc/ot-efr32-dev:latest
options: --user 1001
needs: docker
continue-on-error: true
permissions:
contents: read
strategy:
fail-fast: false
matrix:
Expand All @@ -59,46 +68,54 @@ jobs:
gcc_extract_dir: arm-gnu-toolchain-12.3.rel1-x86_64-arm-none-eabi

steps:
- uses: actions/checkout@v4
with:
submodules: true

- name: Create LFS file hash list
run: git -C third_party/silabs/gecko_sdk lfs ls-files -l | cut -d' ' -f1 | sort > .lfs-assets-id

- name: Restore gecko_sdk LFS cache
uses: actions/cache@v4
id: lfs-cache
with:
path: .git/modules/third_party/silabs/gecko_sdk/lfs
key: lfs-${{ hashFiles('.lfs-assets-id') }}

- name: Git LFS Pull
run: git -C third_party/silabs/gecko_sdk lfs pull

- name: Bootstrap ARM Toolchain
run: |
script/bootstrap arm_toolchain ${{ matrix.gcc_download_url }} ${{ matrix.gcc_extract_dir }}
- name: Build
run: |
export PATH=${HOME}/.local/${{ matrix.gcc_extract_dir }}/bin:$PATH
script/test
- name: Gather SLC generated files
if: failure()
run: |
rm -rf artifact && mkdir artifact
for b in build/*/slc; do
board=$(basename $(dirname "${b}"))
echo "Artifacting '${board}'"
mkdir -p "artifact/${board}"
mv "build/${board}/slc" "artifact/${board}"
done
- uses: actions/upload-artifact@v4
if: failure()
with:
name: build-${{ matrix.gcc_ver }}
path: artifact
- uses: actions/checkout@v4
with:
submodules: true

- name: Create LFS file hash list
run: git -C third_party/silabs/gecko_sdk lfs ls-files -l | cut -d' ' -f1 | sort > .lfs-assets-id

- name: Restore gecko_sdk LFS cache
uses: actions/cache@v4
id: lfs-cache
with:
path: .git/modules/third_party/silabs/gecko_sdk/lfs
key: lfs-${{ hashFiles('.lfs-assets-id') }}

- name: Git LFS Pull
run: git -C third_party/silabs/gecko_sdk lfs pull

- name: Download Docker image
uses: actions/download-artifact@v4
with:
name: ot-efr32-dev

- name: Load docker image
run: docker load -i ot-efr32-dev.tar

- name: Start ot-efr32-dev container
run: docker run --rm -it --user $(id -u) --name ot-efr32-dev -d -v $PWD:/ot-efr32 -w /ot-efr32 ${{ env.DOCKER_IMAGE_SHA_TAG }}

- name: Bootstrap ARM Toolchain
run: docker exec ot-efr32-dev script/bootstrap arm_toolchain ${{ matrix.gcc_download_url }} ${{ matrix.gcc_extract_dir }} ${{ env.TOOLCHAIN_DIR }}

- name: Build
run: docker exec ot-efr32-dev bash -c 'PATH=${{ env.TOOLCHAIN_DIR }}/${{ matrix.gcc_extract_dir }}/bin:$PATH script/test'

- name: Gather SLC generated files
if: failure()
run: |
rm -rf artifact && mkdir artifact
for b in build/*/slc; do
board=$(basename $(dirname "${b}"))
echo "Artifacting '${board}'"
mkdir -p "artifact/${board}"
mv "build/${board}/slc" "artifact/${board}"
done
- uses: actions/upload-artifact@v4
if: failure()
with:
name: build-${{ matrix.gcc_ver }}
path: artifact
126 changes: 59 additions & 67 deletions .github/workflows/docker.yml
Original file line number Diff line number Diff line change
Expand Up @@ -28,108 +28,100 @@

name: Docker

env:
TEST_TAG: siliconlabsinc/ot-efr32-dev:test
SHA_TAG: siliconlabsinc/ot-efr32-dev:${{ github.sha }}
LATEST_TAG: siliconlabsinc/ot-efr32-dev:latest

on:
push:
branches-ignore:
- 'dependabot/**'
pull_request:
branches:
- 'main'

concurrency:
group: ${{ github.workflow }}-${{ github.event.pull_request.number || (github.repository == 'openthread/ot-efr32' && github.run_id) || github.ref }}
cancel-in-progress: true
workflow_call:

permissions: # added using https://github.com/step-security/secure-workflows
contents: read
env:
DOCKER_IMAGE_SHA_TAG: siliconlabsinc/ot-efr32-dev:${{ github.sha }}
DOCKER_IMAGE_LATEST_TAG: siliconlabsinc/ot-efr32-dev:latest

jobs:
metadata:
uses: ./.github/workflows/metadata.yml

build:
name: Build Docker Image
name: Build
runs-on: ubuntu-22.04
needs: metadata
permissions:
contents: read
steps:
- name: Harden Runner
uses: step-security/harden-runner@17d0e2bd7d51742c71671bd19fa12bdc9d40a3d6 # v2.8.1
with:
egress-policy: audit # TODO: change to 'egress-policy: block' after couple of runs

- uses: actions/checkout@v4 # v3.3.0
- uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4.1.7
with:
submodules: true

- name: Set up Docker Buildx
uses: docker/setup-buildx-action@d70bba72b1f3fd22344832f00baa16ece964efeb # v3.3.0
with:
platforms: linux/amd64

- name: Build and export to Docker context
uses: docker/build-push-action@v5
uses: docker/build-push-action@v6
with:
build-args: |
BUILD_DATE=${{ needs.metadata.outputs.date }}
context: .
file: docker/Dockerfile
load: true
build-args: |
- BUILD_DATE=$(date -u +'%Y-%m-%dT%H:%M:%SZ')
platforms:
linux/amd64
tags: |
${{ env.TEST_TAG }}
${{ env.DOCKER_IMAGE_SHA_TAG }}
cache-from: type=gha
cache-to: type=gha,mode=max

- name: Inspect Docker image
run: docker inspect ${{ env.DOCKER_IMAGE_SHA_TAG }}

- name: Container image sanity checks
run: |
# Download container-structure-test
curl -LO https://storage.googleapis.com/container-structure-test/latest/container-structure-test-linux-amd64 \
&& chmod +x container-structure-test-linux-amd64
# Run tests
./container-structure-test-linux-amd64 test --config docker/test-ot-efr32-dev.yml --image ${{ env.TEST_TAG }}
./container-structure-test-linux-amd64 test --config docker/test-ot-efr32-dev.yml --image ${{ env.DOCKER_IMAGE_SHA_TAG }}
- name: Create LFS file hash list
run: git -C third_party/silabs/gecko_sdk lfs ls-files -l | cut -d' ' -f1 | sort > .lfs-assets-id
- name: Export Docker image
run: docker save -o ot-efr32-dev.tar ${{ env.DOCKER_IMAGE_SHA_TAG }}

- name: Restore gecko_sdk LFS cache
uses: actions/cache@v4
id: lfs-cache
- name: Upload Docker image
uses: actions/upload-artifact@v4
with:
path: .git/modules/third_party/silabs/gecko_sdk/lfs
key: lfs-${{ hashFiles('.lfs-assets-id') }}
name: ot-efr32-dev
path: ot-efr32-dev.tar

- name: Git LFS Pull
run: git -C third_party/silabs/gecko_sdk lfs pull

- name: Test build inside container
run: |
docker run -v ${{ github.workspace }}:/ot-efr32/ --user $(id -u) --rm ${{ env.TEST_TAG }} script/build --skip-silabs-apps brd4151a
- name: Login to DockerHub
uses: docker/login-action@0d4c9c5ea7693da7b068278f7b52bda2a190a446 # v3.2.0
if: |
success() &&
github.repository == 'SiliconLabs/ot-efr32' &&
github.event_name != 'pull_request' &&
github.ref == 'refs/heads/main'
with:
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}

- name: Build and push
uses: docker/build-push-action@v5
if: |
success() &&
github.repository == 'SiliconLabs/ot-efr32' &&
github.event_name != 'pull_request' &&
github.ref == 'refs/heads/main'
with:
context: .
file: docker/Dockerfile
push: true
tags: |
${{ env.LATEST_TAG }}
${{ env.SHA_TAG }}
platforms:
linux/amd64
publish-dockerhub:
name: Tag `latest` and publish to DockerHub
runs-on: ubuntu-22.04
needs: [metadata, build]
if: |
github.repository == 'SiliconLabs/ot-efr32' &&
github.event_name != 'pull_request' &&
github.ref == 'refs/heads/main'
steps:
- name: Login to DockerHub
if: github.ref == 'refs/heads/main'
uses: docker/login-action@0d4c9c5ea7693da7b068278f7b52bda2a190a446 # v3.2.0
with:
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}

- name: Download Docker image
uses: actions/download-artifact@v4
with:
name: ot-efr32-dev

- name: Load Docker image
run: docker load -i ot-efr32-dev.tar

- name: Tag Docker image
run: |
docker tag ${{ env.DOCKER_IMAGE_SHA_TAG }} ${{ env.DOCKER_IMAGE_LATEST_TAG }}
- name: Push Docker image
run: |
docker push ${{ env.DOCKER_IMAGE_LATEST_TAG }}
55 changes: 55 additions & 0 deletions .github/workflows/metadata.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
#
# Copyright (c) 2024, The OpenThread Authors.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
# 1. Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# 3. Neither the name of the copyright holder nor the
# names of its contributors may be used to endorse or promote products
# derived from this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
#

name: Metadata

on:
workflow_call:
outputs:
owner:
description: 'The lowercase GitHub repository owner'
value: ${{ jobs.metadata.outputs.owner }}
date:
description: 'The current date'
value: ${{ jobs.metadata.outputs.date }}

jobs:
metadata:
name: Generate required metadata
runs-on: ubuntu-22.04
outputs:
owner: ${{ steps.lowercase_owner.outputs.owner }}
date: ${{ steps.date.outputs.date }}
steps:
- name: Lowercase GitHub repository owner
id: lowercase_owner
run: echo "owner=$(echo ${{ github.repository_owner }} | tr '[:upper:]' '[:lower:]')" >> ${GITHUB_OUTPUT}
- name: Get current date
id: date
run: echo "date=$(date -u +'%Y-%m-%dT%H:%M:%SZ')" >> ${GITHUB_OUTPUT}

2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -76,7 +76,9 @@ __pycache__
.history/

# slc-cli
?/
slc/slc_cli/
slc/**/*.zip

.cache
.toolchain
15 changes: 13 additions & 2 deletions docker/Dockerfile
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
FROM ubuntu:22.04
ARG BASE_IMAGE='ubuntu:22.04'

FROM ${BASE_IMAGE} AS base

ENV TZ="America/New_York"
ENV repo_dir="/ot-efr32"
Expand All @@ -23,7 +25,11 @@ RUN ./script/bootstrap arm_toolchain
COPY ./requirements.txt .
RUN ./script/bootstrap python

# ==============================================================================

# Label the build date before downloading slc to force slc to always be downloaded during a docker build

FROM base AS ot-efr32-dev
ARG BUILD_DATE
LABEL build_date=${BUILD_DATE}

Expand All @@ -34,8 +40,13 @@ RUN mkdir ${SLC_INSTALL_DIR} && \
./script/bootstrap silabs


# Change workdir to root temporarily
WORKDIR /

# Clone repo for convenience
ARG REPO_URL="https://github.com/openthread/ot-efr32"
WORKDIR /
ENV repo_dir="/ot-efr32"
RUN rm -rf ${repo_dir} && git clone ${REPO_URL} ${repo_dir}

# Change workdir back to repo
WORKDIR ${repo_dir}
Loading

0 comments on commit 8d94119

Please sign in to comment.