2020-12-23 18:39:02 +05:00
|
|
|
#!/usr/bin/env bash
|
|
|
|
|
|
|
|
# The Docker Image Mirror script.
|
2022-06-28 20:15:13 +05:00
|
|
|
# Designed to mirror amd64 and arm64 images somewhere you want.
|
|
|
|
# Required environment variables:
|
|
|
|
#
|
|
|
|
# * REGISTRY: to which registry images should be sent. Should not contain protocols,
|
|
|
|
# should be the very same as you use 'docker login' with.
|
|
|
|
# * REGISTRY_USERNAME: user name to use for destination registry.
|
|
|
|
# * REGISTRY_PASSWORD: password to use for destination registry's user.
|
|
|
|
# * REGISTRY_PROJECT: path within registry (or it's part) where images will be push'd.
|
|
|
|
# Should start with '/'!
|
|
|
|
|
|
|
|
# What images we will mirror. Collected by executing scripts in "images" directory, see
|
|
|
|
# "collect_images" function.
|
2020-12-23 18:39:02 +05:00
|
|
|
WHAT_TO_MIRROR=()
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
# New tag for currently processing image as it will be used with 'docker push'.
|
|
|
|
# Updates for every image. Contains only path, image name and base tag from source.
|
|
|
|
REMOTE_IMAGE=""
|
|
|
|
|
|
|
|
# Is image multiarch? E.g. is we're successfully fetched not only amd64 image,
|
|
|
|
# but also an arm64? Updates for every image.
|
|
|
|
MULTIARCH=0
|
|
|
|
|
|
|
|
function cleanup() {
|
|
|
|
local image=$1
|
|
|
|
|
|
|
|
docker image rm "${image}" &> /dev/null
|
|
|
|
docker image rm "${REMOTE_IMAGE}" &> /dev/null
|
|
|
|
docker image rm "${REMOTE_IMAGE}"-amd64 &> /dev/null
|
|
|
|
docker image rm "${REMOTE_IMAGE}"-arm64 &> /dev/null
|
|
|
|
}
|
|
|
|
|
|
|
|
function collect_images() {
|
|
|
|
# Load shell files and execute them to get list of mirrorred images.
|
|
|
|
MIRROR_CONFIGS=$(ls ./images/*.sh)
|
|
|
|
# shellcheck disable=SC2068
|
|
|
|
for file in ${MIRROR_CONFIGS[@]}; do
|
|
|
|
echo "Importing ${file}..."
|
|
|
|
# shellcheck disable=SC2086,SC2207,SC2206
|
|
|
|
WHAT_TO_MIRROR=( ${WHAT_TO_MIRROR[@]} $(bash ${file}) )
|
|
|
|
done
|
|
|
|
|
|
|
|
echo "Images to mirror: ${WHAT_TO_MIRROR[*]}"
|
|
|
|
}
|
|
|
|
|
|
|
|
function get_amd64_image() {
|
2022-06-27 12:53:36 +05:00
|
|
|
local image=$1
|
2022-06-28 20:15:13 +05:00
|
|
|
|
2020-12-23 18:39:02 +05:00
|
|
|
image_name=$(echo "${image}" | cut -d":" -f 1)
|
|
|
|
image_version=$(echo "${image}" | cut -d":" -f 2)
|
|
|
|
|
2022-06-28 22:10:56 +05:00
|
|
|
# Check if amd64 layers should be fetched.
|
|
|
|
if docker manifest inspect "${REMOTE_IMAGE}-amd64" &> /dev/null; then
|
|
|
|
echo -e "\t* Layers for amd64 architecture for this image exist."
|
|
|
|
|
2022-06-29 15:21:43 +05:00
|
|
|
return 2
|
2022-06-28 22:10:56 +05:00
|
|
|
fi
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
echo -ne "\t* Getting amd64 layers... "
|
2022-06-27 13:07:04 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
if ! docker pull --platform=linux/amd64 "${image}" &> /dev/null; then
|
|
|
|
echo "FAIL!"
|
2022-06-27 13:07:04 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
return 1
|
2020-12-23 18:39:02 +05:00
|
|
|
fi
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
echo -n "Downloaded, "
|
2022-06-27 13:07:04 +05:00
|
|
|
|
2022-06-28 20:16:10 +05:00
|
|
|
# shellcheck disable=SC1083
|
2022-06-28 20:15:13 +05:00
|
|
|
image_hash=$(docker images -a | grep "^${image_name}" | grep "${image_version}" | awk {' print $3 '})
|
2020-12-23 18:39:02 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
if ! docker tag "${image_hash}" "${REMOTE_IMAGE}-amd64" &> /dev/null; then
|
|
|
|
echo "but tagging failed!"
|
2022-06-27 00:08:44 +05:00
|
|
|
exit 1
|
2020-12-23 18:39:02 +05:00
|
|
|
fi
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
echo "tagged."
|
|
|
|
}
|
|
|
|
|
|
|
|
function get_arm64_image() {
|
|
|
|
local image=$1
|
|
|
|
|
|
|
|
image_name=$(echo "${image}" | cut -d":" -f 1)
|
|
|
|
image_version=$(echo "${image}" | cut -d":" -f 2)
|
|
|
|
|
2022-06-28 22:10:56 +05:00
|
|
|
# Check if arm64 layers should be fetched.
|
|
|
|
if docker manifest inspect "${REMOTE_IMAGE}-arm64" &> /dev/null; then
|
|
|
|
echo -e "\t* Layers for arm64 architecture for this image exist."
|
|
|
|
|
2022-06-29 15:21:43 +05:00
|
|
|
return 2
|
2022-06-28 22:10:56 +05:00
|
|
|
fi
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
echo -ne "\t* Getting arm64 layers... "
|
|
|
|
|
|
|
|
# arm64 layers might be missing. So we just put "FAIL!" here and do nothing else.
|
|
|
|
if ! docker pull --platform=linux/arm64 "${image}" &> /dev/null; then
|
|
|
|
echo "FAIL!"
|
|
|
|
|
2022-06-29 15:21:43 +05:00
|
|
|
return 1
|
2022-06-28 20:15:13 +05:00
|
|
|
fi
|
|
|
|
|
|
|
|
echo -n "Downloaded, "
|
|
|
|
|
2022-06-28 20:16:10 +05:00
|
|
|
# shellcheck disable=SC1083
|
2022-06-28 20:15:13 +05:00
|
|
|
image_hash=$(docker images -a | grep "^${image_name}" | grep "${image_version}" | awk {' print $3 '})
|
|
|
|
|
|
|
|
if ! docker tag "${image_hash}" "${REMOTE_IMAGE}-arm64" &> /dev/null; then
|
|
|
|
echo "but tagging failed!"
|
2022-06-27 00:08:44 +05:00
|
|
|
exit 1
|
2020-12-23 18:39:02 +05:00
|
|
|
fi
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
echo "tagged."
|
2020-12-23 18:39:02 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
MULTIARCH=1
|
2020-12-23 18:39:02 +05:00
|
|
|
}
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
function login_to_registry() {
|
|
|
|
# Login to registry.
|
2022-06-28 20:25:11 +05:00
|
|
|
echo "Logging into '${REGISTRY}' as '${REGISTRY_USERNAME}'..."
|
2022-06-28 20:21:13 +05:00
|
|
|
docker login -u "${REGISTRY_USERNAME}" -p "${REGISTRY_PASSWORD}" "${REGISTRY}"
|
2022-06-28 20:15:13 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
function mirror() {
|
|
|
|
local image=$1
|
|
|
|
image_name=$(echo "${image}" | cut -d":" -f 1)
|
|
|
|
image_version=$(echo "${image}" | cut -d":" -f 2)
|
|
|
|
|
|
|
|
echo "* Mirroring ${image}"
|
2022-06-27 12:33:03 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
REMOTE_IMAGE="${REGISTRY}${REGISTRY_PROJECT}/${image}"
|
|
|
|
MULTIARCH=0
|
|
|
|
|
|
|
|
# We presumes that amd64 image should always be available.
|
|
|
|
if ! get_amd64_image "${image}"; then
|
|
|
|
echo "! Image mirroring failed! Cannot obtain amd64 image!"
|
2022-06-27 12:33:03 +05:00
|
|
|
fi
|
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
get_arm64_image "${image}"
|
|
|
|
push_multiarch_image "${image}"
|
|
|
|
cleanup "${image}"
|
|
|
|
}
|
|
|
|
|
|
|
|
function push_multiarch_image() {
|
|
|
|
local image=$1
|
|
|
|
|
2022-06-28 20:25:11 +05:00
|
|
|
if ! docker push "${REMOTE_IMAGE}"-amd64 &> /dev/null; then
|
2022-06-28 20:15:13 +05:00
|
|
|
echo -e "\t! amd64 image push failed!"
|
2022-06-28 20:25:11 +05:00
|
|
|
else
|
|
|
|
echo -e "\t* amd64 image pushed"
|
2022-06-28 20:15:13 +05:00
|
|
|
fi
|
|
|
|
|
2022-06-28 20:25:11 +05:00
|
|
|
if ! docker push "${REMOTE_IMAGE}"-arm64 &> /dev/null; then
|
2022-06-28 20:15:13 +05:00
|
|
|
echo -e "\t! arm64 image push failed!"
|
2022-06-28 20:25:11 +05:00
|
|
|
else
|
|
|
|
echo -e "\t* arm64 image pushed"
|
2022-06-28 20:15:13 +05:00
|
|
|
fi
|
2022-06-27 12:33:03 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
if [ ${MULTIARCH} -eq 1 ]; then
|
|
|
|
echo -e "\t* Image is multi-arch, creating and pushing a manifest..."
|
|
|
|
docker manifest create "${REMOTE_IMAGE}" \
|
|
|
|
--amend "${REMOTE_IMAGE}"-amd64 \
|
|
|
|
--amend "${REMOTE_IMAGE}"-arm64 \
|
|
|
|
&> /dev/null
|
|
|
|
|
|
|
|
docker manifest push "${REMOTE_IMAGE}" &> /dev/null
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
function start_docker_daemon() {
|
|
|
|
# Starting Docker daemon.
|
2022-07-30 00:02:30 +05:00
|
|
|
/usr/local/bin/dockerd --data-root=/var/lib/docker --max-concurrent-uploads 1 &
|
2022-06-28 20:15:13 +05:00
|
|
|
|
|
|
|
# Wait for it.
|
|
|
|
echo "Waiting for Docker daemon to start..."
|
|
|
|
while true; do
|
|
|
|
if docker ps &> /dev/null; then
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
}
|
2022-06-27 12:33:03 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
start_docker_daemon
|
|
|
|
login_to_registry
|
|
|
|
collect_images
|
2022-06-26 23:53:26 +05:00
|
|
|
|
2022-06-28 20:15:13 +05:00
|
|
|
for package in "${WHAT_TO_MIRROR[@]}"; do
|
|
|
|
mirror "${package}"
|
2020-12-23 18:39:02 +05:00
|
|
|
done
|