5.4 KiB
ADR 0020: Docker Build Strategy - Traditional vs Buildx
Status
Accepted ✅
Context
The DanceLessonsCoach CI/CD pipeline initially used Docker Buildx (docker buildx build --push) for building and pushing Docker cache images. However, this approach encountered several issues:
Issues with Buildx Approach
- TLS Certificate Problems: Buildx had difficulty with self-signed certificates, requiring complex workaround steps
- Performance Concerns: Buildx setup and execution was significantly slower than expected
- Complexity: Buildx introduced additional complexity without providing immediate benefits
- Reliability Issues: Buildx builds were less reliable in the GitHub Actions environment
Working Solution Analysis
The working webapp CI/CD pipeline uses traditional docker build + docker push approach:
# Working approach from webapp
- name: Build and push image to Gitea Container Registry
run: |-
docker build -t app .
docker tag app gitea.arcodange.lab/${{ github.repository }}:$TAG
docker push gitea.arcodange.lab/${{ github.repository }}:$TAG
This approach is simpler, more reliable, and works consistently with self-signed certificates.
Decision
Replace Docker Buildx with traditional docker build + push for the CI/CD pipeline.
Implementation
# New approach
- name: Build and push Docker cache image
if: steps.check_cache.outputs.cache_hit == 'false'
run: |
IMAGE_NAME="${{ env.CI_REGISTRY }}/${{ env.GITEA_ORG }}/${{ env.GITEA_REPO }}-build-cache:${{ steps.calculate_hash.outputs.deps_hash }}"
echo "Building cache image: $IMAGE_NAME"
# Build the image using traditional docker build
docker build \
--file Dockerfile.build \
--tag "$IMAGE_NAME" \
.
# Push the image
docker push "$IMAGE_NAME"
echo "✅ Build cache image pushed successfully"
Benefits
- Simplicity: Traditional approach is easier to understand and debug
- Reliability: Consistent behavior across different environments
- Certificate Handling: Works seamlessly with self-signed certificates
- Performance: Faster execution without Buildx overhead
- Compatibility: Better compatibility with GitHub Actions environment
Trade-offs
What We Lose
- Multi-platform builds: Cannot build for multiple architectures simultaneously
- BuildKit caching: Less sophisticated caching mechanism
- Advanced features: No secret mounting, SSH agents, etc.
- Parallel processing: Slower builds without Buildx optimizations
What We Gain
- Stability: More reliable CI/CD pipeline
- Simplicity: Easier to maintain and troubleshoot
- Consistency: Matches proven patterns from working projects
- Faster feedback: Quicker build times in practice
Rationale
- Current Needs: We don't need multi-platform builds or advanced BuildKit features
- Simple Dockerfile: Our
Dockerfile.builddoesn't require Buildx-specific features - Proven Pattern: Traditional approach works reliably in production (webapp project)
- CI Stability: Reliability is more important than advanced features for CI/CD
Future Considerations
When to Reconsider Buildx
- Multi-platform needs: If we need ARM/AMD64 builds simultaneously
- Complex builds: If Dockerfile requires BuildKit-specific features
- Performance optimization: If build times become unacceptable
- Certificate issues resolved: If Docker Buildx improves self-signed certificate handling
Migration Path
If we need to reintroduce Buildx in the future:
- Fix certificate issues properly at the Docker daemon level
- Test thoroughly in staging environment
- Monitor performance impact
- Document benefits clearly for the specific use case
Alternatives Considered
Option 1: Keep Buildx with Certificate Workaround
- ❌ Complex setup with questionable reliability
- ❌ Slow performance in GitHub Actions
- ❌ Ongoing maintenance burden
Option 2: Use Insecure Registry Flag
docker buildx build --allow security.insecure --push .
- ❌ Security concerns
- ❌ Not recommended for production
- ❌ Temporary workaround, not solution
Option 3: Traditional Docker Build + Push ✅ CHOSEN
- ✅ Simple and reliable
- ✅ Proven in production
- ✅ Better performance in practice
- ✅ Easy to maintain
Decision Outcome
Chosen Option: Traditional docker build + push (Option 3)
This decision prioritizes CI/CD reliability and simplicity over advanced features we don't currently need. The traditional approach has been proven to work consistently in our environment and matches the successful pattern from the webapp project.
Success Metrics
- CI/CD reliability: No TLS certificate failures
- Build consistency: Predictable build times
- Maintenance: Reduced complexity and debugging time
- Compatibility: Works across all target environments
References
- Docker Buildx Documentation
- Docker Build Documentation
- GitHub Actions Docker Examples
- webapp CI/CD Pipeline
Approved by: @arcodange Date: 2026-04-07 Supersedes: None Superseded by: None