docker-publish-beta.sh 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. #!/bin/bash
  2. # Build and push multi-architecture Docker image to GHCR (private beta)
  3. #
  4. # Usage:
  5. # ./docker-publish-beta.sh [version] [--parallel]
  6. #
  7. # Examples:
  8. # ./docker-publish-beta.sh 0.2.0b # Sequential build
  9. # ./docker-publish-beta.sh 0.2.0b --parallel # Build both archs simultaneously
  10. #
  11. # All versions are also tagged as 'beta'
  12. #
  13. # Prerequisites:
  14. # 1. Log in to ghcr.io:
  15. # echo $GITHUB_TOKEN | docker login ghcr.io -u YOUR_USERNAME --password-stdin
  16. #
  17. # 2. Create a GitHub Personal Access Token with 'write:packages' scope:
  18. # https://github.com/settings/tokens/new?scopes=write:packages
  19. #
  20. # 3. After first push, set package to Private in GitHub → Packages → Settings
  21. # and add beta testers via Manage Access
  22. #
  23. # Supported architectures:
  24. # - linux/amd64 (x86_64, most servers/desktops)
  25. # - linux/arm64 (Raspberry Pi 4/5, Apple Silicon via emulation)
  26. set -e
  27. # Configuration
  28. GHCR_REGISTRY="ghcr.io"
  29. IMAGE_NAME="maziggy/bambuddy-beta"
  30. GHCR_IMAGE="${GHCR_REGISTRY}/${IMAGE_NAME}"
  31. PLATFORMS="linux/amd64,linux/arm64"
  32. BUILDER_NAME="bambuddy-builder"
  33. # Colors for output
  34. RED='\033[0;31m'
  35. GREEN='\033[0;32m'
  36. YELLOW='\033[1;33m'
  37. BLUE='\033[0;34m'
  38. NC='\033[0m' # No Color
  39. # Parse arguments
  40. VERSION=""
  41. PARALLEL=false
  42. for arg in "$@"; do
  43. case $arg in
  44. --parallel)
  45. PARALLEL=true
  46. ;;
  47. *)
  48. if [ -z "$VERSION" ]; then
  49. VERSION="$arg"
  50. fi
  51. ;;
  52. esac
  53. done
  54. if [ -z "$VERSION" ]; then
  55. echo -e "${YELLOW}Usage: $0 <version> [--parallel]${NC}"
  56. echo "Example: $0 0.2.0b"
  57. echo " $0 0.2.0b --parallel # Build both architectures simultaneously"
  58. exit 1
  59. fi
  60. # Get CPU count
  61. CPU_COUNT=$(nproc 2>/dev/null || sysctl -n hw.ncpu 2>/dev/null || echo 4)
  62. echo -e "${GREEN}================================================${NC}"
  63. echo -e "${GREEN} Building multi-arch BETA image${NC}"
  64. echo -e "${GREEN} Version: ${VERSION}${NC}"
  65. echo -e "${GREEN} Platforms: ${PLATFORMS}${NC}"
  66. echo -e "${GREEN} CPU cores: ${CPU_COUNT}${NC}"
  67. if [ "$PARALLEL" = true ]; then
  68. echo -e "${GREEN} Mode: PARALLEL (both archs simultaneously)${NC}"
  69. else
  70. echo -e "${GREEN} Mode: Sequential (amd64 → arm64)${NC}"
  71. fi
  72. echo -e "${GREEN} Registry: ${GHCR_IMAGE}${NC}"
  73. echo -e "${GREEN}================================================${NC}"
  74. echo ""
  75. # Check registry login
  76. if ! grep -q "ghcr.io" ~/.docker/config.json 2>/dev/null; then
  77. echo -e "${YELLOW}Warning: You may not be logged in to ghcr.io${NC}"
  78. echo "Run: echo \$GITHUB_TOKEN | docker login ghcr.io -u YOUR_USERNAME --password-stdin"
  79. echo ""
  80. fi
  81. # Setup buildx builder if not exists
  82. echo -e "${BLUE}[1/4] Setting up Docker Buildx...${NC}"
  83. if ! docker buildx inspect "$BUILDER_NAME" >/dev/null 2>&1; then
  84. echo "Creating new buildx builder: $BUILDER_NAME (optimized for ${CPU_COUNT} cores)"
  85. docker buildx create \
  86. --name "$BUILDER_NAME" \
  87. --driver docker-container \
  88. --driver-opt network=host \
  89. --driver-opt "env.BUILDKIT_STEP_LOG_MAX_SIZE=10000000" \
  90. --buildkitd-flags "--allow-insecure-entitlement network.host --oci-worker-gc=false" \
  91. --config /dev/stdin <<EOF
  92. [worker.oci]
  93. max-parallelism = ${CPU_COUNT}
  94. EOF
  95. docker buildx inspect --bootstrap "$BUILDER_NAME"
  96. fi
  97. docker buildx use "$BUILDER_NAME"
  98. # Verify builder supports multi-platform
  99. echo -e "${BLUE}[2/4] Verifying multi-platform support...${NC}"
  100. if ! docker buildx inspect --bootstrap | grep -q "linux/arm64"; then
  101. echo -e "${YELLOW}Installing QEMU for cross-platform builds...${NC}"
  102. docker run --privileged --rm tonistiigi/binfmt --install all
  103. fi
  104. # Build tags — version + beta (not latest)
  105. TAGS="-t ${GHCR_IMAGE}:${VERSION} -t ${GHCR_IMAGE}:beta"
  106. echo -e "${BLUE}[3/4] Building and pushing...${NC}"
  107. # Common build args (no cache to ensure clean builds)
  108. BUILD_ARGS="--provenance=false --sbom=false --no-cache --pull"
  109. if [ "$PARALLEL" = true ]; then
  110. # Parallel build: Build each architecture separately then combine manifests
  111. echo -e "${YELLOW}Building amd64 and arm64 in parallel (${CPU_COUNT} cores each, no cache)...${NC}"
  112. # Build amd64 in background
  113. (
  114. echo -e "${BLUE}[amd64] Starting build...${NC}"
  115. docker buildx build \
  116. --platform linux/amd64 \
  117. -t "${GHCR_IMAGE}:${VERSION}-amd64" \
  118. ${BUILD_ARGS} \
  119. --push \
  120. . 2>&1 | sed 's/^/[amd64] /'
  121. echo -e "${GREEN}[amd64] Complete!${NC}"
  122. ) &
  123. PID_AMD64=$!
  124. # Build arm64 in background
  125. (
  126. echo -e "${BLUE}[arm64] Starting build...${NC}"
  127. docker buildx build \
  128. --platform linux/arm64 \
  129. -t "${GHCR_IMAGE}:${VERSION}-arm64" \
  130. ${BUILD_ARGS} \
  131. --push \
  132. . 2>&1 | sed 's/^/[arm64] /'
  133. echo -e "${GREEN}[arm64] Complete!${NC}"
  134. ) &
  135. PID_ARM64=$!
  136. # Wait for both builds
  137. echo "Waiting for parallel builds to complete..."
  138. wait $PID_AMD64
  139. wait $PID_ARM64
  140. # Create multi-arch manifest
  141. echo -e "${BLUE}Creating multi-arch manifest...${NC}"
  142. docker buildx imagetools create \
  143. -t "${GHCR_IMAGE}:${VERSION}" -t "${GHCR_IMAGE}:beta" \
  144. "${GHCR_IMAGE}:${VERSION}-amd64" \
  145. "${GHCR_IMAGE}:${VERSION}-arm64"
  146. else
  147. # Sequential build (default): Build both platforms in one command
  148. echo -e "${YELLOW}Building sequentially with ${CPU_COUNT} cores (no cache)...${NC}"
  149. DOCKER_BUILDKIT=1 docker buildx build \
  150. --platform "$PLATFORMS" \
  151. ${BUILD_ARGS} \
  152. $TAGS \
  153. --push \
  154. .
  155. fi
  156. echo -e "${BLUE}[4/4] Verifying manifest...${NC}"
  157. docker buildx imagetools inspect "${GHCR_IMAGE}:${VERSION}"
  158. echo ""
  159. echo -e "${GREEN}================================================${NC}"
  160. echo -e "${GREEN} Successfully pushed multi-arch BETA image:${NC}"
  161. echo -e "${GREEN}================================================${NC}"
  162. echo " ${GHCR_IMAGE}:${VERSION}"
  163. echo " ${GHCR_IMAGE}:beta"
  164. echo ""
  165. echo -e "${BLUE}Supported platforms:${NC}"
  166. echo " - linux/amd64 (Intel/AMD servers, desktops)"
  167. echo " - linux/arm64 (Raspberry Pi 4/5, Apple Silicon)"
  168. echo ""
  169. echo -e "${GREEN}Beta testers can run:${NC}"
  170. echo " docker pull ${GHCR_IMAGE}:${VERSION}"
  171. echo " docker pull ${GHCR_IMAGE}:beta"
  172. echo ""
  173. echo -e "${YELLOW}Reminder: Set package to Private in GitHub → Packages → Settings${NC}"