docker-publish.sh 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. #!/bin/bash
  2. # Build and push multi-architecture Docker image to GitHub Container Registry AND Docker Hub
  3. #
  4. # Usage:
  5. # ./docker-publish.sh [version] [--parallel] [--ghcr-only] [--dockerhub-only]
  6. #
  7. # Examples:
  8. # ./docker-publish.sh 0.1.9b # Sequential build, push to both registries
  9. # ./docker-publish.sh 0.1.9b --parallel # Build both archs simultaneously
  10. # ./docker-publish.sh 0.1.9b --ghcr-only # Only push to GHCR
  11. # ./docker-publish.sh 0.1.9b --dockerhub-only # Only push to Docker Hub
  12. #
  13. # Note: Stable versions are also tagged as 'latest'. Beta versions (ending in 'b') are not.
  14. #
  15. # Prerequisites:
  16. # 1. Log in to ghcr.io:
  17. # echo $GITHUB_TOKEN | docker login ghcr.io -u YOUR_USERNAME --password-stdin
  18. #
  19. # 2. Log in to Docker Hub:
  20. # docker login -u YOUR_USERNAME
  21. #
  22. # 3. Create a GitHub Personal Access Token with 'write:packages' scope:
  23. # https://github.com/settings/tokens/new?scopes=write:packages
  24. #
  25. # Supported architectures:
  26. # - linux/amd64 (x86_64, most servers/desktops)
  27. # - linux/arm64 (Raspberry Pi 4/5, Apple Silicon via emulation)
  28. set -e
  29. # Configuration
  30. GHCR_REGISTRY="ghcr.io"
  31. DOCKERHUB_REGISTRY="docker.io"
  32. IMAGE_NAME="maziggy/bambuddy"
  33. GHCR_IMAGE="${GHCR_REGISTRY}/${IMAGE_NAME}"
  34. DOCKERHUB_IMAGE="${DOCKERHUB_REGISTRY}/${IMAGE_NAME}"
  35. PLATFORMS="linux/amd64,linux/arm64"
  36. BUILDER_NAME="bambuddy-builder"
  37. # Colors for output
  38. RED='\033[0;31m'
  39. GREEN='\033[0;32m'
  40. YELLOW='\033[1;33m'
  41. BLUE='\033[0;34m'
  42. NC='\033[0m' # No Color
  43. # Parse arguments
  44. VERSION=""
  45. PARALLEL=false
  46. PUSH_GHCR=true
  47. PUSH_DOCKERHUB=true
  48. for arg in "$@"; do
  49. case $arg in
  50. --parallel)
  51. PARALLEL=true
  52. ;;
  53. --ghcr-only)
  54. PUSH_DOCKERHUB=false
  55. ;;
  56. --dockerhub-only)
  57. PUSH_GHCR=false
  58. ;;
  59. *)
  60. if [ -z "$VERSION" ]; then
  61. VERSION="$arg"
  62. fi
  63. ;;
  64. esac
  65. done
  66. if [ -z "$VERSION" ]; then
  67. echo -e "${YELLOW}Usage: $0 <version> [--parallel] [--ghcr-only] [--dockerhub-only]${NC}"
  68. echo "Example: $0 0.1.9b"
  69. echo " $0 0.1.9b --parallel # Build both architectures simultaneously"
  70. echo " $0 0.1.9b --ghcr-only # Only push to GitHub Container Registry"
  71. echo " $0 0.1.9b --dockerhub-only # Only push to Docker Hub"
  72. exit 1
  73. fi
  74. # Get CPU count
  75. CPU_COUNT=$(nproc 2>/dev/null || sysctl -n hw.ncpu 2>/dev/null || echo 4)
  76. echo -e "${GREEN}================================================${NC}"
  77. echo -e "${GREEN} Building multi-arch image${NC}"
  78. echo -e "${GREEN} Version: ${VERSION}${NC}"
  79. echo -e "${GREEN} Platforms: ${PLATFORMS}${NC}"
  80. echo -e "${GREEN} CPU cores: ${CPU_COUNT}${NC}"
  81. if [ "$PARALLEL" = true ]; then
  82. echo -e "${GREEN} Mode: PARALLEL (both archs simultaneously)${NC}"
  83. else
  84. echo -e "${GREEN} Mode: Sequential (amd64 → arm64)${NC}"
  85. fi
  86. echo -e "${GREEN} Registries:${NC}"
  87. if [ "$PUSH_GHCR" = true ]; then
  88. echo -e "${GREEN} - ${GHCR_IMAGE}${NC}"
  89. fi
  90. if [ "$PUSH_DOCKERHUB" = true ]; then
  91. echo -e "${GREEN} - ${DOCKERHUB_IMAGE}${NC}"
  92. fi
  93. echo -e "${GREEN}================================================${NC}"
  94. echo ""
  95. # Check registry logins
  96. if [ "$PUSH_GHCR" = true ]; then
  97. if ! grep -q "ghcr.io" ~/.docker/config.json 2>/dev/null; then
  98. echo -e "${YELLOW}Warning: You may not be logged in to ghcr.io${NC}"
  99. echo "Run: echo \$GITHUB_TOKEN | docker login ghcr.io -u YOUR_USERNAME --password-stdin"
  100. echo ""
  101. fi
  102. fi
  103. if [ "$PUSH_DOCKERHUB" = true ]; then
  104. if ! grep -q "index.docker.io\|docker.io" ~/.docker/config.json 2>/dev/null; then
  105. echo -e "${RED}Error: You are not logged in to Docker Hub${NC}"
  106. echo "Run: docker login -u YOUR_USERNAME"
  107. echo ""
  108. exit 1
  109. fi
  110. fi
  111. # Setup buildx builder if not exists
  112. echo -e "${BLUE}[1/4] Setting up Docker Buildx...${NC}"
  113. if ! docker buildx inspect "$BUILDER_NAME" >/dev/null 2>&1; then
  114. echo "Creating new buildx builder: $BUILDER_NAME (optimized for ${CPU_COUNT} cores)"
  115. docker buildx create \
  116. --name "$BUILDER_NAME" \
  117. --driver docker-container \
  118. --driver-opt network=host \
  119. --driver-opt "env.BUILDKIT_STEP_LOG_MAX_SIZE=10000000" \
  120. --buildkitd-flags "--allow-insecure-entitlement network.host --oci-worker-gc=false" \
  121. --config /dev/stdin <<EOF
  122. [worker.oci]
  123. max-parallelism = ${CPU_COUNT}
  124. EOF
  125. docker buildx inspect --bootstrap "$BUILDER_NAME"
  126. fi
  127. docker buildx use "$BUILDER_NAME"
  128. # Verify builder supports multi-platform
  129. echo -e "${BLUE}[2/4] Verifying multi-platform support...${NC}"
  130. if ! docker buildx inspect --bootstrap | grep -q "linux/arm64"; then
  131. echo -e "${YELLOW}Installing QEMU for cross-platform builds...${NC}"
  132. docker run --privileged --rm tonistiigi/binfmt --install all
  133. fi
  134. # Only tag as 'latest' for stable releases (not beta versions ending in 'b')
  135. TAG_LATEST=true
  136. if [[ "$VERSION" =~ b[0-9]*$ ]]; then
  137. TAG_LATEST=false
  138. echo -e "${YELLOW}Beta version detected — skipping 'latest' tag${NC}"
  139. fi
  140. # Build tags for all target registries
  141. TAGS=""
  142. if [ "$PUSH_GHCR" = true ]; then
  143. TAGS="$TAGS -t ${GHCR_IMAGE}:${VERSION}"
  144. [ "$TAG_LATEST" = true ] && TAGS="$TAGS -t ${GHCR_IMAGE}:latest"
  145. fi
  146. if [ "$PUSH_DOCKERHUB" = true ]; then
  147. TAGS="$TAGS -t ${DOCKERHUB_IMAGE}:${VERSION}"
  148. [ "$TAG_LATEST" = true ] && TAGS="$TAGS -t ${DOCKERHUB_IMAGE}:latest"
  149. fi
  150. echo -e "${BLUE}[3/4] Building and pushing...${NC}"
  151. # Common build args (no cache to ensure clean builds)
  152. BUILD_ARGS="--provenance=false --sbom=false --no-cache --pull"
  153. if [ "$PARALLEL" = true ]; then
  154. # Parallel build: Build each architecture separately then combine manifests
  155. echo -e "${YELLOW}Building amd64 and arm64 in parallel (${CPU_COUNT} cores each, no cache)...${NC}"
  156. # Build per-arch staging tags for each target registry
  157. ARCH_TAGS_AMD64=""
  158. ARCH_TAGS_ARM64=""
  159. if [ "$PUSH_GHCR" = true ]; then
  160. ARCH_TAGS_AMD64="$ARCH_TAGS_AMD64 -t ${GHCR_IMAGE}:${VERSION}-amd64"
  161. ARCH_TAGS_ARM64="$ARCH_TAGS_ARM64 -t ${GHCR_IMAGE}:${VERSION}-arm64"
  162. fi
  163. if [ "$PUSH_DOCKERHUB" = true ]; then
  164. ARCH_TAGS_AMD64="$ARCH_TAGS_AMD64 -t ${DOCKERHUB_IMAGE}:${VERSION}-amd64"
  165. ARCH_TAGS_ARM64="$ARCH_TAGS_ARM64 -t ${DOCKERHUB_IMAGE}:${VERSION}-arm64"
  166. fi
  167. # Build amd64 in background
  168. (
  169. echo -e "${BLUE}[amd64] Starting build...${NC}"
  170. docker buildx build \
  171. --platform linux/amd64 \
  172. ${ARCH_TAGS_AMD64} \
  173. ${BUILD_ARGS} \
  174. --push \
  175. . 2>&1 | sed 's/^/[amd64] /'
  176. echo -e "${GREEN}[amd64] Complete!${NC}"
  177. ) &
  178. PID_AMD64=$!
  179. # Build arm64 in background
  180. (
  181. echo -e "${BLUE}[arm64] Starting build...${NC}"
  182. docker buildx build \
  183. --platform linux/arm64 \
  184. ${ARCH_TAGS_ARM64} \
  185. ${BUILD_ARGS} \
  186. --push \
  187. . 2>&1 | sed 's/^/[arm64] /'
  188. echo -e "${GREEN}[arm64] Complete!${NC}"
  189. ) &
  190. PID_ARM64=$!
  191. # Wait for both builds
  192. echo "Waiting for parallel builds to complete..."
  193. wait $PID_AMD64
  194. wait $PID_ARM64
  195. # Create multi-arch manifests per registry (no cross-registry blob copies)
  196. echo -e "${BLUE}Creating multi-arch manifests...${NC}"
  197. if [ "$PUSH_GHCR" = true ]; then
  198. echo -e "${BLUE} Creating GHCR manifest...${NC}"
  199. GHCR_MANIFEST_TAGS="-t ${GHCR_IMAGE}:${VERSION}"
  200. [ "$TAG_LATEST" = true ] && GHCR_MANIFEST_TAGS="$GHCR_MANIFEST_TAGS -t ${GHCR_IMAGE}:latest"
  201. docker buildx imagetools create \
  202. $GHCR_MANIFEST_TAGS \
  203. "${GHCR_IMAGE}:${VERSION}-amd64" \
  204. "${GHCR_IMAGE}:${VERSION}-arm64"
  205. fi
  206. if [ "$PUSH_DOCKERHUB" = true ]; then
  207. echo -e "${BLUE} Creating Docker Hub manifest...${NC}"
  208. DH_MANIFEST_TAGS="-t ${DOCKERHUB_IMAGE}:${VERSION}"
  209. [ "$TAG_LATEST" = true ] && DH_MANIFEST_TAGS="$DH_MANIFEST_TAGS -t ${DOCKERHUB_IMAGE}:latest"
  210. docker buildx imagetools create \
  211. $DH_MANIFEST_TAGS \
  212. "${DOCKERHUB_IMAGE}:${VERSION}-amd64" \
  213. "${DOCKERHUB_IMAGE}:${VERSION}-arm64"
  214. fi
  215. else
  216. # Sequential build (default): Build both platforms in one command
  217. echo -e "${YELLOW}Building sequentially with ${CPU_COUNT} cores (no cache)...${NC}"
  218. DOCKER_BUILDKIT=1 docker buildx build \
  219. --platform "$PLATFORMS" \
  220. ${BUILD_ARGS} \
  221. $TAGS \
  222. --push \
  223. .
  224. fi
  225. echo -e "${BLUE}[4/4] Verifying manifests...${NC}"
  226. if [ "$PUSH_GHCR" = true ]; then
  227. echo -e "${BLUE}GHCR:${NC}"
  228. docker buildx imagetools inspect "${GHCR_IMAGE}:${VERSION}"
  229. fi
  230. if [ "$PUSH_DOCKERHUB" = true ]; then
  231. echo -e "${BLUE}Docker Hub:${NC}"
  232. docker buildx imagetools inspect "${DOCKERHUB_IMAGE}:${VERSION}"
  233. fi
  234. echo ""
  235. echo -e "${GREEN}================================================${NC}"
  236. echo -e "${GREEN} Successfully pushed multi-arch image:${NC}"
  237. echo -e "${GREEN}================================================${NC}"
  238. if [ "$PUSH_GHCR" = true ]; then
  239. echo " GHCR:"
  240. echo " - ${GHCR_IMAGE}:${VERSION}"
  241. [ "$TAG_LATEST" = true ] && echo " - ${GHCR_IMAGE}:latest"
  242. fi
  243. if [ "$PUSH_DOCKERHUB" = true ]; then
  244. echo " Docker Hub:"
  245. echo " - ${DOCKERHUB_IMAGE}:${VERSION}"
  246. [ "$TAG_LATEST" = true ] && echo " - ${DOCKERHUB_IMAGE}:latest"
  247. fi
  248. echo ""
  249. echo -e "${BLUE}Supported platforms:${NC}"
  250. echo " - linux/amd64 (Intel/AMD servers, desktops)"
  251. echo " - linux/arm64 (Raspberry Pi 4/5, Apple Silicon)"
  252. echo ""
  253. echo -e "${GREEN}Users can now run:${NC}"
  254. if [ "$PUSH_GHCR" = true ]; then
  255. echo " docker pull ${GHCR_IMAGE}:${VERSION}"
  256. fi
  257. if [ "$PUSH_DOCKERHUB" = true ]; then
  258. echo " docker pull ${DOCKERHUB_IMAGE}:${VERSION}"
  259. echo " docker pull ${IMAGE_NAME}:${VERSION} # shorthand"
  260. fi