|
1 | | -name: buildx |
| 1 | +name: Docker Build |
2 | 2 |
|
3 | 3 | on: |
4 | 4 | pull_request: |
5 | 5 | branches: main |
6 | 6 | push: |
7 | | - branches: main |
| 7 | + branches: |
| 8 | + - main |
| 9 | + - dev |
8 | 10 | tags: |
9 | 11 | - v* |
10 | 12 |
|
| 13 | +# Cancel in-progress builds for same branch/PR |
| 14 | +concurrency: |
| 15 | + group: ${{ github.workflow }}-${{ github.ref }} |
| 16 | + cancel-in-progress: true |
| 17 | + |
| 18 | +# Minimal permissions for GITHUB_TOKEN |
| 19 | +permissions: |
| 20 | + contents: read |
| 21 | + packages: write |
| 22 | + attestations: write |
| 23 | + id-token: write |
| 24 | + |
11 | 25 | jobs: |
12 | | - buildx: |
| 26 | + build: |
13 | 27 | runs-on: ubuntu-latest |
14 | 28 | steps: |
15 | | - - |
16 | | - name: Checkout |
17 | | - uses: actions/checkout@v4 |
18 | | - - |
19 | | - name: Prepare |
20 | | - id: prepare |
21 | | - run: | |
22 | | - DOCKER_IMAGE=portableprogrammer/status-light |
23 | | - DOCKER_PLATFORMS=linux/amd64,linux/arm/v7,linux/arm64 |
24 | | - VERSION=edge |
| 29 | + - name: Checkout |
| 30 | + uses: actions/checkout@v4 # 2026-02-05: v4.2.2 |
25 | 31 |
|
26 | | - if [[ $GITHUB_REF == refs/tags/* ]]; then |
27 | | - VERSION=${GITHUB_REF#refs/tags/v} |
28 | | - fi |
| 32 | + - name: Docker metadata |
| 33 | + id: meta |
| 34 | + uses: docker/metadata-action@v5 # 2026-02-05: v5.7.0 |
| 35 | + with: |
| 36 | + images: portableprogrammer/status-light |
| 37 | + tags: | |
| 38 | + # Tag PRs with 'edge' |
| 39 | + type=edge,branch=main |
| 40 | + # Tag releases with semver |
| 41 | + type=semver,pattern={{version}} |
| 42 | + type=semver,pattern={{major}}.{{minor}} |
| 43 | + type=semver,pattern={{major}},enable=${{ !startsWith(github.ref, 'refs/tags/v0.') }} |
| 44 | + labels: | |
| 45 | + org.opencontainers.image.title=Status-Light |
| 46 | + org.opencontainers.image.description=Multi-platform presence status indicator for smart RGB lights |
| 47 | + org.opencontainers.image.vendor=PortableProgrammer |
| 48 | + org.opencontainers.image.licenses=MIT |
29 | 49 |
|
30 | | - TAGS="--tag ${DOCKER_IMAGE}:${VERSION}" |
31 | | - if [[ $VERSION =~ ^[0-9]{1,3}\.[0-9]{1,3}(\.[0-9]{1,3})?$ ]]; then |
32 | | - TAGS="$TAGS --tag ${DOCKER_IMAGE}:latest" |
33 | | - fi |
| 50 | + - name: Set up QEMU |
| 51 | + uses: docker/setup-qemu-action@v3 # 2026-02-05: v3.2.0 |
34 | 52 |
|
35 | | - echo "docker_image=${DOCKER_IMAGE}" >> "$GITHUB_ENV" |
36 | | - echo "version=${VERSION}" >> "$GITHUB_ENV" |
37 | | - echo "buildx_args=--platform ${DOCKER_PLATFORMS} --build-arg VERSION=${VERSION} --build-arg BUILD_DATE=$(date -u +'%Y-%m-%dT%H:%M:%SZ') --build-arg VCS_REF=${GITHUB_SHA::8} ${TAGS} --file ./Dockerfiles/Dockerfile ./" >> "$GITHUB_ENV" |
38 | | - - |
39 | | - name: Set up QEMU |
40 | | - uses: docker/setup-qemu-action@v3 |
41 | | - - |
42 | | - name: Set up Docker Buildx |
43 | | - uses: docker/setup-buildx-action@v3 |
| 53 | + - name: Set up Docker Buildx |
| 54 | + uses: docker/setup-buildx-action@v3 # 2026-02-05: v3.7.1 |
| 55 | + |
| 56 | + - name: Docker Hub login |
| 57 | + if: github.event_name != 'pull_request' |
| 58 | + uses: docker/login-action@v3 # 2026-02-05: v3.3.0 |
44 | 59 | with: |
45 | | - # Enable Docker layer caching |
46 | | - buildkitd-flags: --debug |
47 | | - - |
48 | | - name: Docker Buildx (build) |
49 | | - run: | |
50 | | - docker buildx build --output "type=image,push=false" ${{ env.buildx_args }} |
51 | | - - |
52 | | - name: Docker Login |
53 | | - if: success() && github.event_name != 'pull_request' |
54 | | - env: |
55 | | - DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }} |
56 | | - DOCKER_PASSWORD: ${{ secrets.DOCKER_PASSWORD }} |
57 | | - run: | |
58 | | - echo "${DOCKER_PASSWORD}" | docker login --username "${DOCKER_USERNAME}" --password-stdin |
59 | | - - |
60 | | - name: Docker Buildx (push) |
61 | | - if: success() && github.event_name != 'pull_request' |
62 | | - run: | |
63 | | - docker buildx build --output "type=image,push=true" ${{ env.buildx_args }} |
64 | | - - |
65 | | - name: Docker Check Manifest |
66 | | - if: always() && github.event_name != 'pull_request' |
| 60 | + username: ${{ secrets.DOCKER_USERNAME }} |
| 61 | + password: ${{ secrets.DOCKER_PASSWORD }} |
| 62 | + |
| 63 | + - name: Build and push |
| 64 | + id: build |
| 65 | + uses: docker/build-push-action@v6 # 2026-02-05: v6.10.0 |
| 66 | + with: |
| 67 | + context: . |
| 68 | + file: ./Dockerfiles/Dockerfile |
| 69 | + platforms: linux/amd64,linux/arm/v7,linux/arm64 |
| 70 | + push: ${{ github.event_name != 'pull_request' }} |
| 71 | + tags: ${{ steps.meta.outputs.tags }} |
| 72 | + labels: ${{ steps.meta.outputs.labels }} |
| 73 | + # GitHub Actions cache for faster builds |
| 74 | + cache-from: type=gha |
| 75 | + cache-to: type=gha,mode=max |
| 76 | + # Supply chain security |
| 77 | + provenance: mode=max |
| 78 | + sbom: true |
| 79 | + |
| 80 | + - name: Docker Hub description |
| 81 | + if: github.event_name != 'pull_request' && startsWith(github.ref, 'refs/tags/v') |
| 82 | + uses: peter-evans/dockerhub-description@v4 # 2026-02-05: v4.0.0 |
| 83 | + with: |
| 84 | + username: ${{ secrets.DOCKER_USERNAME }} |
| 85 | + password: ${{ secrets.DOCKER_PASSWORD }} |
| 86 | + repository: portableprogrammer/status-light |
| 87 | + short-description: Multi-platform presence status indicator for smart RGB lights |
| 88 | + continue-on-error: true |
| 89 | + |
| 90 | + - name: Verify multi-arch manifest |
| 91 | + if: github.event_name != 'pull_request' |
67 | 92 | run: | |
68 | | - docker run --rm mplatform/mquery ${{ env.docker_image }}:${{ env.version }} |
69 | | - - |
70 | | - name: Clear |
71 | | - if: always() && github.event_name != 'pull_request' |
| 93 | + docker buildx imagetools inspect ${{ steps.meta.outputs.tags }} | tee manifest.txt |
| 94 | + echo "## Multi-Architecture Manifest" >> $GITHUB_STEP_SUMMARY |
| 95 | + echo '```' >> $GITHUB_STEP_SUMMARY |
| 96 | + cat manifest.txt >> $GITHUB_STEP_SUMMARY |
| 97 | + echo '```' >> $GITHUB_STEP_SUMMARY |
| 98 | +
|
| 99 | + - name: Build summary |
| 100 | + if: always() |
72 | 101 | run: | |
73 | | - rm -f ${HOME}/.docker/config.json |
| 102 | + echo "## Build Summary" >> $GITHUB_STEP_SUMMARY |
| 103 | + echo "- **Event:** ${{ github.event_name }}" >> $GITHUB_STEP_SUMMARY |
| 104 | + echo "- **Ref:** ${{ github.ref }}" >> $GITHUB_STEP_SUMMARY |
| 105 | + echo "- **Platforms:** linux/amd64, linux/arm/v7, linux/arm64" >> $GITHUB_STEP_SUMMARY |
| 106 | + echo "- **Tags:** ${{ steps.meta.outputs.tags }}" >> $GITHUB_STEP_SUMMARY |
| 107 | + if [[ "${{ github.event_name }}" != "pull_request" ]]; then |
| 108 | + echo "- **Digest:** ${{ steps.build.outputs.digest }}" >> $GITHUB_STEP_SUMMARY |
| 109 | + echo "- **Pushed:** ✅ Yes" >> $GITHUB_STEP_SUMMARY |
| 110 | + else |
| 111 | + echo "- **Pushed:** ⏭️ Skipped (PR)" >> $GITHUB_STEP_SUMMARY |
| 112 | + fi |
0 commit comments