174e8562aa
Encountered a similar problem as in https://github.com/mostlygeek/llama-swap/issues/709 but in my case I only needed the :cpu version. So decided to add the github action to build arm64 combined with the amd64 version on the same :cpu tag. Already tested it from this fork: ghcr.io/rhtenhove/llama-swap:cpu and it works perfectly fine. Adding GPU support is a whole other beast, needing quite a bit more work and isn't something I can test.
215 lines
7.7 KiB
Bash
Executable File
215 lines
7.7 KiB
Bash
Executable File
#!/bin/bash
|
|
|
|
set -euo pipefail
|
|
|
|
cd $(dirname "$0")
|
|
|
|
# use this to test locally, example:
|
|
# GITHUB_TOKEN=$(gh auth token) LOG_DEBUG=1 DEBUG_ABORT_BUILD=1 ./docker/build-container.sh rocm
|
|
# you need read:package scope on the token. Generate a personal access token with
|
|
# the scopes: gist, read:org, repo, write:packages
|
|
# then: gh auth login (and copy/paste the new token)
|
|
|
|
LOG_DEBUG=${LOG_DEBUG:-0}
|
|
DEBUG_ABORT_BUILD=${DEBUG_ABORT_BUILD:-}
|
|
|
|
log_debug() {
|
|
if [ "$LOG_DEBUG" = "1" ]; then
|
|
echo "[DEBUG] $*"
|
|
fi
|
|
}
|
|
|
|
log_info() {
|
|
echo "[INFO] $*"
|
|
}
|
|
|
|
ARCH=$1
|
|
PUSH_IMAGES=${2:-false}
|
|
|
|
# List of allowed architectures
|
|
ALLOWED_ARCHS=("intel" "vulkan" "musa" "cuda" "cuda13" "cpu" "rocm")
|
|
|
|
# Check if ARCH is in the allowed list
|
|
if [[ ! " ${ALLOWED_ARCHS[@]} " =~ " ${ARCH} " ]]; then
|
|
log_info "Error: ARCH must be one of the following: ${ALLOWED_ARCHS[@]}"
|
|
exit 1
|
|
fi
|
|
|
|
# Check if GITHUB_TOKEN is set and not empty
|
|
if [[ -z "${GITHUB_TOKEN:-}" ]]; then
|
|
log_info "Error: GITHUB_TOKEN is not set or is empty."
|
|
exit 1
|
|
fi
|
|
|
|
# Set llama.cpp base image, customizable using the BASE_LLAMACPP_IMAGE environment
|
|
# variable, this permits testing with forked llama.cpp repositories
|
|
BASE_IMAGE=${BASE_LLAMACPP_IMAGE:-ghcr.io/ggml-org/llama.cpp}
|
|
SD_IMAGE=${BASE_SDCPP_IMAGE:-ghcr.io/leejet/stable-diffusion.cpp}
|
|
|
|
# LS_REPO is the destination of the built container image — defaults to the
|
|
# current GitHub repository so forked CI builds publish to the fork's own
|
|
# ghcr.io namespace without code changes.
|
|
LS_REPO=${GITHUB_REPOSITORY:-mostlygeek/llama-swap}
|
|
|
|
# LS_BINARY_REPO is where the llama-swap release tarball is downloaded
|
|
# from. Decoupled from LS_REPO so forks (which usually have no releases of
|
|
# their own) can still build a container by pulling the canonical binary
|
|
# from upstream. Override via the LS_BINARY_REPO env var when you maintain
|
|
# fork-side releases.
|
|
LS_BINARY_REPO=${LS_BINARY_REPO:-mostlygeek/llama-swap}
|
|
|
|
# the most recent llama-swap tag
|
|
# have to strip out the 'v' due to .tar.gz file naming.
|
|
# Authenticated request — unauth'd github.com API is 60/hr per IP and GHA
|
|
# runners share IPs, so the call regularly returns rate-limit JSON and
|
|
# `.tag_name` then resolves to "null", producing a bogus `vnull` URL below.
|
|
LS_VER=$(curl -s -H "Authorization: Bearer $GITHUB_TOKEN" \
|
|
"https://api.github.com/repos/${LS_BINARY_REPO}/releases/latest" \
|
|
| jq -r .tag_name | sed 's/v//')
|
|
|
|
if [[ -z "$LS_VER" || "$LS_VER" == "null" ]]; then
|
|
log_info "Error: could not resolve latest llama-swap release tag from ${LS_BINARY_REPO}"
|
|
exit 1
|
|
fi
|
|
|
|
# Fetches the most recent llama.cpp tag matching the given prefix
|
|
# Handles pagination to search beyond the first 100 results
|
|
# $1 - tag_prefix (e.g., "server" or "server-vulkan")
|
|
# Returns: the version number extracted from the tag
|
|
fetch_llama_tag() {
|
|
local tag_prefix=$1
|
|
local page=1
|
|
local per_page=100
|
|
|
|
while true; do
|
|
log_debug "Fetching page $page for tag prefix: $tag_prefix"
|
|
|
|
local response=$(curl -s -H "Authorization: Bearer $GITHUB_TOKEN" \
|
|
"https://api.github.com/users/ggml-org/packages/container/llama.cpp/versions?per_page=${per_page}&page=${page}")
|
|
|
|
# Check for API errors
|
|
if echo "$response" | jq -e '.message' > /dev/null 2>&1; then
|
|
local error_msg=$(echo "$response" | jq -r '.message')
|
|
log_info "GitHub API error: $error_msg"
|
|
return 1
|
|
fi
|
|
|
|
# Check if response is empty array (no more pages)
|
|
if [ "$(echo "$response" | jq 'length')" -eq 0 ]; then
|
|
log_debug "No more pages (empty response)"
|
|
return 1
|
|
fi
|
|
|
|
# Extract matching tag from this page
|
|
local found_tag=$(echo "$response" | jq -r \
|
|
".[] | select(.metadata.container.tags[]? | startswith(\"$tag_prefix\")) | .metadata.container.tags[] | select(startswith(\"$tag_prefix\"))" \
|
|
| sort -r | head -n1)
|
|
|
|
if [ -n "$found_tag" ]; then
|
|
log_debug "Found tag: $found_tag on page $page"
|
|
echo "$found_tag" | awk -F '-' '{print $NF}'
|
|
return 0
|
|
fi
|
|
|
|
page=$((page + 1))
|
|
|
|
# Safety limit to prevent infinite loops
|
|
if [ $page -gt 50 ]; then
|
|
log_info "Reached pagination safety limit (50 pages)"
|
|
return 1
|
|
fi
|
|
done
|
|
}
|
|
|
|
if [ "$ARCH" == "cpu" ]; then
|
|
LCPP_TAG=$(fetch_llama_tag "server")
|
|
BASE_TAG=server-${LCPP_TAG}
|
|
else
|
|
LCPP_TAG=$(fetch_llama_tag "server-${ARCH}")
|
|
BASE_TAG=server-${ARCH}-${LCPP_TAG}
|
|
fi
|
|
|
|
SD_TAG=master-${ARCH}
|
|
|
|
# Abort if LCPP_TAG is empty.
|
|
if [[ -z "$LCPP_TAG" ]]; then
|
|
log_info "Abort: Could not find llama-server container for arch: $ARCH"
|
|
exit 1
|
|
else
|
|
log_info "LCPP_TAG: $LCPP_TAG"
|
|
fi
|
|
|
|
if [[ ! -z "$DEBUG_ABORT_BUILD" ]]; then
|
|
log_info "Abort: DEBUG_ABORT_BUILD set"
|
|
exit 0
|
|
fi
|
|
|
|
# cpu is the only backend with a multi-arch upstream base
|
|
# (ghcr.io/ggml-org/llama.cpp:server-bXXXX ships amd64+arm64); GPU backends
|
|
# are amd64-only and stay on the original `docker build` path so the
|
|
# sd-server layer can still FROM the just-built image via the local
|
|
# dockerd image store (buildx's container driver has a separate store
|
|
# that doesn't share with dockerd, which breaks the sd build).
|
|
if [ "$ARCH" == "cpu" ]; then
|
|
if [ "$PUSH_IMAGES" == "true" ]; then
|
|
BUILDX_FLAGS="--push --platform linux/amd64,linux/arm64"
|
|
else
|
|
# Smoke build: validate both platforms but emit no output. buildx
|
|
# on the docker-container driver defaults to cacheonly when
|
|
# neither --push nor --load is given, so each arch fully builds
|
|
# and a regression in either fails CI — without materializing the
|
|
# image or needing to --load (which is multi-arch-incompatible).
|
|
BUILDX_FLAGS="--platform linux/amd64,linux/arm64"
|
|
fi
|
|
fi
|
|
|
|
for CONTAINER_TYPE in non-root root; do
|
|
CONTAINER_TAG="ghcr.io/${LS_REPO}:v${LS_VER}-${ARCH}-${LCPP_TAG}"
|
|
CONTAINER_LATEST="ghcr.io/${LS_REPO}:${ARCH}"
|
|
USER_UID=0
|
|
USER_GID=0
|
|
USER_HOME=/root
|
|
|
|
if [ "$CONTAINER_TYPE" == "non-root" ]; then
|
|
CONTAINER_TAG="${CONTAINER_TAG}-non-root"
|
|
CONTAINER_LATEST="${CONTAINER_LATEST}-non-root"
|
|
USER_UID=10001
|
|
USER_GID=10001
|
|
USER_HOME=/app
|
|
fi
|
|
|
|
log_info "Building $CONTAINER_TYPE $CONTAINER_TAG $LS_VER"
|
|
if [ "$ARCH" == "cpu" ]; then
|
|
docker buildx build $BUILDX_FLAGS --provenance=false \
|
|
-f llama-swap.Containerfile \
|
|
--build-arg BASE_TAG=${BASE_TAG} --build-arg LS_VER=${LS_VER} --build-arg UID=${USER_UID} \
|
|
--build-arg LS_REPO=${LS_BINARY_REPO} --build-arg GID=${USER_GID} --build-arg USER_HOME=${USER_HOME} \
|
|
--build-arg BASE_IMAGE=${BASE_IMAGE} \
|
|
-t ${CONTAINER_TAG} -t ${CONTAINER_LATEST} .
|
|
else
|
|
docker build --provenance=false -f llama-swap.Containerfile \
|
|
--build-arg BASE_TAG=${BASE_TAG} --build-arg LS_VER=${LS_VER} --build-arg UID=${USER_UID} \
|
|
--build-arg LS_REPO=${LS_BINARY_REPO} --build-arg GID=${USER_GID} --build-arg USER_HOME=${USER_HOME} \
|
|
-t ${CONTAINER_TAG} -t ${CONTAINER_LATEST} \
|
|
--build-arg BASE_IMAGE=${BASE_IMAGE} .
|
|
fi
|
|
|
|
# For architectures with stable-diffusion.cpp support, layer sd-server on top.
|
|
# Stays on `docker build` so the base resolves from local dockerd.
|
|
case "$ARCH" in
|
|
"musa" | "vulkan")
|
|
log_info "Adding sd-server to $CONTAINER_TAG"
|
|
docker build --provenance=false -f llama-swap-sd.Containerfile \
|
|
--build-arg BASE=${CONTAINER_TAG} \
|
|
--build-arg SD_IMAGE=${SD_IMAGE} --build-arg SD_TAG=${SD_TAG} \
|
|
--build-arg UID=${USER_UID} --build-arg GID=${USER_GID} \
|
|
-t ${CONTAINER_TAG} -t ${CONTAINER_LATEST} . ;;
|
|
esac
|
|
|
|
# cpu builds push inline via buildx --push; all other archs push here.
|
|
if [ "$ARCH" != "cpu" ] && [ "$PUSH_IMAGES" == "true" ]; then
|
|
docker push ${CONTAINER_TAG}
|
|
docker push ${CONTAINER_LATEST}
|
|
fi
|
|
done
|