2023-02-15 12:16:31 +00:00
#!/usr/bin/env bash
2023-08-28 19:24:23 +00:00
VERSION = "v0.2.5"
### ChangeNotes: Added an -s option to include stopped contianers in the check.
2023-02-02 21:02:41 +00:00
Github = "https://github.com/mag37/dockcheck"
2023-03-05 20:25:22 +00:00
RawUrl = "https://raw.githubusercontent.com/mag37/dockcheck/main/dockcheck.sh"
2023-02-02 21:02:41 +00:00
2023-03-04 18:56:26 +00:00
### Variables for self updating
ScriptArgs = ( " $@ " )
ScriptPath = " $( readlink -f " $0 " ) "
ScriptName = " $( basename " $ScriptPath " ) "
ScriptWorkDir = " $( dirname " $ScriptPath " ) "
2023-01-18 10:50:00 +00:00
2023-03-05 19:44:05 +00:00
### Check if there's a new release of the script:
LatestRelease = " $( curl -s -r 0-50 $RawUrl | sed -n "/VERSION/s/VERSION=//p" | tr -d '"' ) "
LatestChanges = " $( curl -s -r 0-200 $RawUrl | sed -n "/ChangeNotes/s/### ChangeNotes: //p" ) "
2023-01-30 19:39:27 +00:00
### Help Function:
2023-01-19 11:09:29 +00:00
Help( ) {
2023-02-09 12:03:27 +00:00
echo "Syntax: dockcheck.sh [OPTION] [part of name to filter]"
2023-02-26 12:16:54 +00:00
echo "Example: dockcheck.sh -a -e nextcloud,heimdall"
2023-02-09 12:03:27 +00:00
echo
echo "Options:"
echo "-h Print this Help."
echo "-a|y Automatic updates, without interaction."
echo "-n No updates, only checking availability."
2023-02-26 06:49:57 +00:00
echo "-e Exclude containers, separated by comma."
2023-02-15 12:16:31 +00:00
echo "-p Auto-Prune dangling images after update."
2023-08-28 19:24:23 +00:00
echo "-r Allow updating images for docker run, wont update the container"
echo "-s Include stopped containers in the check. (Logic: docker ps -a)"
2023-02-09 12:03:27 +00:00
}
2023-01-19 11:09:29 +00:00
2023-08-28 19:24:23 +00:00
Stopped = ""
while getopts "aynprhse:" options; do
2023-02-09 12:03:27 +00:00
case " ${ options } " in
a| y) UpdYes = "yes" ; ;
n) UpdYes = "no" ; ;
2023-02-10 20:06:12 +00:00
r) DrUp = "yes" ; ;
2023-02-15 12:16:31 +00:00
p) PruneQ = "yes" ; ;
2023-02-26 06:49:57 +00:00
e) Exclude = ${ OPTARG } ; ;
2023-08-28 19:24:23 +00:00
s) Stopped = "-a" ; ;
2023-02-09 12:03:27 +00:00
h| *) Help ; exit 0 ; ;
esac
done
shift " $(( OPTIND-1)) "
2023-01-19 11:09:29 +00:00
2023-03-04 18:56:26 +00:00
self_update_git( ) {
cd " $ScriptWorkDir " || { printf "Path error, skipping update.\n" ; return ; }
[ [ $( builtin type -P git) ] ] || { printf "Git not installed, skipping update.\n" ; return ; }
2023-03-05 20:59:08 +00:00
ScriptUpstream = $( git rev-parse --abbrev-ref --symbolic-full-name "@{upstream}" ) || { printf "Script not in git directory, choose a different method.\n" ; self_update_select ; return ; }
2023-03-04 18:56:26 +00:00
git fetch
2023-03-04 19:21:31 +00:00
[ -n " $( git diff --name-only " $ScriptUpstream " " $ScriptName " ) " ] && {
2023-03-04 18:56:26 +00:00
printf "%s\n" "Pulling the latest version."
2023-03-04 19:31:20 +00:00
# git checkout "$ScriptUpstream"
2023-03-04 18:56:26 +00:00
git pull --force
2023-03-05 12:50:28 +00:00
printf "%s\n" "--- starting over with the updated version ---"
2023-03-04 18:56:26 +00:00
cd - || { printf "Path error.\n" ; return ; }
exec " $ScriptPath " " ${ ScriptArgs [@] } " # run the new script with old arguments
exit 1 # exit the old instance
}
echo "Local is already latest."
}
self_update_curl( ) {
cp " $ScriptPath " " $ScriptPath " .bak
if [ [ $( builtin type -P curl) ] ] ; then
curl -L $RawUrl > " $ScriptPath " ; chmod +x " $ScriptPath "
printf "%s\n" "--- starting over with the updated version ---"
exec " $ScriptPath " " ${ ScriptArgs [@] } " # run the new script with old arguments
exit 1 # exit the old instance
else
printf "curl not available - download the update manually: %s \n" " $RawUrl "
fi
}
2023-03-05 13:08:56 +00:00
self_update_select( ) {
2023-03-04 18:56:26 +00:00
read -r -p "Choose update procedure (or do it manually) - git/curl/[no]: " SelfUpQ
if [ [ " $SelfUpQ " = = "git" ] ] ; then self_update_git ;
elif [ [ " $SelfUpQ " = = "curl" ] ] ; then self_update_curl ;
else printf "Download it manually from the repo: %s \n\n" " $Github "
fi
2023-03-05 13:08:56 +00:00
}
### Choose from list -function:
choosecontainers( ) {
while [ [ -z " $ChoiceClean " ] ] ; do
read -r -p "Enter number(s) separated by comma, [a] for all - [q] to quit: " Choice
if [ [ " $Choice " = ~ [ qQnN] ] ] ; then
exit 0
elif [ [ " $Choice " = ~ [ aAyY] ] ] ; then
SelectedUpdates = ( " ${ GotUpdates [@] } " )
ChoiceClean = ${ Choice //[,. : ;]/ }
else
ChoiceClean = ${ Choice //[,. : ;]/ }
for CC in $ChoiceClean ; do
if [ [ " $CC " -lt 1 || " $CC " -gt $UpdCount ] ] ; then # reset choice if out of bounds
echo " Number not in list: $CC " ; unset ChoiceClean ; break 1
else
SelectedUpdates += ( " ${ GotUpdates [ $CC -1] } " )
fi
done
fi
done
printf "\nUpdating containers:\n"
printf "%s\n" " ${ SelectedUpdates [@] } "
printf "\n"
}
### Version check & initiate self update
2023-06-28 16:03:56 +00:00
[ [ " $VERSION " != " $LatestRelease " ] ] && { printf "New version available! Local: %s - Latest: %s \n Change Notes: %s \n" " $VERSION " " $LatestRelease " " $LatestChanges " ; [ [ -z " $UpdYes " ] ] && self_update_select ; }
2023-03-04 18:56:26 +00:00
2023-02-09 12:03:27 +00:00
### Set $1 to a variable for name filtering later.
SearchName = " $1 "
2023-02-26 06:49:57 +00:00
### Create array of excludes
IFS = ',' read -r -a Excludes <<< " $Exclude " ; unset IFS
2023-01-19 11:09:29 +00:00
2023-02-09 12:03:27 +00:00
### Check if required binary exists in PATH or directory:
2023-02-15 12:16:31 +00:00
if [ [ $( builtin type -P "regctl" ) ] ] ; then regbin = "regctl" ;
elif [ [ -f "./regctl" ] ] ; then regbin = "./regctl" ;
2023-11-19 19:20:05 +00:00
elif [ [ -f " $ScriptWorkDir /regctl " ] ] ; then regbin = " $ScriptWorkDir /regctl " ;
2023-02-09 12:03:27 +00:00
else
2023-02-15 12:16:31 +00:00
read -r -p "Required dependency 'regctl' missing, do you want it downloaded? y/[n] " GetDep
if [ [ " $GetDep " = ~ [ yY] ] ] ; then
2023-02-09 12:03:27 +00:00
### Check arch:
case " $( uname --machine) " in
x86_64| amd64) architecture = "amd64" ; ;
arm64| aarch64) architecture = "arm64" ; ;
2023-02-15 12:16:31 +00:00
*) echo "Architecture not supported, exiting." ; exit 1; ;
2023-02-09 12:03:27 +00:00
esac
2023-02-15 12:16:31 +00:00
RegUrl = " https://github.com/regclient/regclient/releases/latest/download/regctl-linux- $architecture "
if [ [ $( builtin type -P curl) ] ] ; then curl -L $RegUrl > ./regctl ; chmod +x ./regctl ; regbin = "./regctl" ;
elif [ [ $( builtin type -P wget) ] ] ; then wget $RegUrl -O ./regctl ; chmod +x ./regctl ; regbin = "./regctl" ;
else
printf "%s\n" "curl/wget not available - get regctl manually from the repo link, quitting."
fi
2023-02-09 12:03:27 +00:00
else
printf "%s\n" "Dependency missing, quitting."
2023-02-15 12:16:31 +00:00
exit 1
2023-02-09 12:03:27 +00:00
fi
fi
2023-02-15 12:16:31 +00:00
### final check if binary is correct
$regbin version & > /dev/null || { printf "%s\n" "regctl is not working - try to remove it and re-download it, exiting." ; exit 1; }
2023-02-09 12:03:27 +00:00
### Check docker compose binary:
2023-03-05 13:18:57 +00:00
if docker compose version & > /dev/null ; then DockerBin = "docker compose" ;
elif docker-compose -v & > /dev/null; then DockerBin = "docker-compose" ;
2023-02-10 20:06:12 +00:00
elif docker -v & > /dev/null; then
printf "%s\n" "No docker compose binary available, using plain docker (Not recommended!)"
printf "%s\n" "'docker run' will ONLY update images, not the container itself."
2023-02-09 12:03:27 +00:00
else
2023-02-10 20:06:12 +00:00
printf "%s\n" "No docker binaries available, exiting."
2023-02-15 12:16:31 +00:00
exit 1
2023-02-09 12:03:27 +00:00
fi
2023-01-30 09:08:13 +00:00
2023-02-09 12:03:27 +00:00
### Numbered List -function:
options( ) {
2023-02-15 12:16:31 +00:00
num = 1
for i in " ${ GotUpdates [@] } " ; do
2023-02-09 12:03:27 +00:00
echo " $num ) $i "
( ( num++) )
done
}
2023-01-30 09:08:13 +00:00
2023-06-21 18:09:31 +00:00
### Listing typed exclusions:
if [ [ -n ${ Excludes [*] } ] ] ; then
printf "\n\033[0;34mExcluding these names:\033[0m\n"
printf "%s\n" " ${ Excludes [@] } "
printf "\n"
fi
2023-02-09 12:03:27 +00:00
### Check the image-hash of every running container VS the registry
2023-08-28 19:24:23 +00:00
for i in $( docker ps $Stopped --filter " name= $SearchName " --format '{{.Names}}' ) ; do
### Looping every item over the list of excluded names and skipping:
2023-06-21 18:09:31 +00:00
for e in " ${ Excludes [@] } " ; do [ [ " $i " = = " $e " ] ] && continue 2 ; done
2023-02-09 12:03:27 +00:00
printf ". "
RepoUrl = $( docker inspect " $i " --format= '{{.Config.Image}}' )
LocalHash = $( docker image inspect " $RepoUrl " --format '{{.RepoDigests}}' )
2023-02-12 18:40:42 +00:00
### Checking for errors while setting the variable:
if RegHash = $( $regbin image digest --list " $RepoUrl " 2>/dev/null) ; then
2023-02-09 12:03:27 +00:00
if [ [ " $LocalHash " = *" $RegHash " * ] ] ; then NoUpdates += ( " $i " ) ; else GotUpdates += ( " $i " ) ; fi
else
GotErrors += ( " $i " )
fi
done
2023-01-18 10:50:00 +00:00
2023-02-09 12:03:27 +00:00
### Sort arrays alphabetically
IFS = $'\n'
NoUpdates = ( $( sort <<< " ${ NoUpdates [*] } " ) )
2023-02-01 18:39:43 +00:00
GotUpdates = ( $( sort <<< " ${ GotUpdates [*] } " ) )
GotErrors = ( $( sort <<< " ${ GotErrors [*] } " ) )
unset IFS
2023-02-15 12:16:31 +00:00
### Define how many updates are available
UpdCount = " ${# GotUpdates [@] } "
2023-01-30 09:08:13 +00:00
2023-01-18 10:50:00 +00:00
### List what containers got updates or not
2023-02-12 18:40:42 +00:00
if [ [ -n ${ NoUpdates [*] } ] ] ; then
2023-02-19 13:21:22 +00:00
printf "\n\033[0;32mContainers on latest version:\033[0m\n"
2023-01-18 10:50:00 +00:00
printf "%s\n" " ${ NoUpdates [@] } "
fi
2023-02-12 18:40:42 +00:00
if [ [ -n ${ GotErrors [*] } ] ] ; then
2023-02-19 13:21:22 +00:00
printf "\n\033[0;31mContainers with errors, wont get updated:\033[0m\n"
2023-01-20 11:47:17 +00:00
printf "%s\n" " ${ GotErrors [@] } "
fi
2023-02-12 18:40:42 +00:00
if [ [ -n ${ GotUpdates [*] } ] ] ; then
2023-02-19 13:21:22 +00:00
printf "\n\033[0;33mContainers with updates available:\033[0m\n"
2023-02-12 18:40:42 +00:00
[ [ -z " $UpdYes " ] ] && options || printf "%s\n" " ${ GotUpdates [@] } "
2023-01-30 09:08:13 +00:00
fi
2023-01-18 10:50:00 +00:00
2023-01-18 20:45:31 +00:00
### Optionally get updates if there's any
if [ -n " $GotUpdates " ] ; then
2023-01-19 11:09:29 +00:00
if [ -z " $UpdYes " ] ; then
2023-02-19 13:21:22 +00:00
printf "\n\033[0;36mChoose what containers to update.\033[0m\n"
2023-02-07 13:56:18 +00:00
choosecontainers
2023-01-30 09:08:13 +00:00
else
SelectedUpdates = ( " ${ GotUpdates [@] } " )
fi
2023-02-07 13:56:18 +00:00
if [ " $UpdYes " = = " ${ UpdYes #[Nn] } " ] ; then
2023-06-27 17:34:20 +00:00
NumberofUpdates = " ${# SelectedUpdates [@] } "
CurrentQue = 0
2023-01-30 09:08:13 +00:00
for i in " ${ SelectedUpdates [@] } "
2023-03-01 19:39:02 +00:00
do
2023-06-27 17:34:20 +00:00
( ( CurrentQue += 1) )
2023-03-01 19:39:02 +00:00
unset CompleteConfs
2023-02-09 10:33:02 +00:00
ContPath = $( docker inspect " $i " --format '{{ index .Config.Labels "com.docker.compose.project.working_dir" }}' )
ContConfigFile = $( docker inspect " $i " --format '{{ index .Config.Labels "com.docker.compose.project.config_files" }}' )
2023-02-04 11:35:08 +00:00
ContName = $( docker inspect " $i " --format '{{ index .Config.Labels "com.docker.compose.service" }}' )
2023-02-19 13:21:22 +00:00
ContEnv = $( docker inspect " $i " --format '{{index .Config.Labels "com.docker.compose.project.environment_file" }}' )
2023-02-23 12:05:06 +00:00
ContImage = $( docker inspect " $i " --format= '{{.Config.Image}}' )
2023-02-10 20:06:12 +00:00
### Checking if compose-values are empty - hence started with docker run:
if [ -z " $ContPath " ] ; then
if [ " $DrUp " = = "yes" ] ; then
2023-02-12 18:40:42 +00:00
docker pull " $ContImage "
2023-02-10 20:06:12 +00:00
printf "%s\n" " $i got a new image downloaded, rebuild manually with preferred 'docker run'-parameters "
else
2023-02-12 18:40:42 +00:00
printf "\n\033[33;1m%s\033[0m has no compose labels, probably started with docker run - \033[33;1mskipping\033[0m\n\n" " $i "
2023-02-10 20:06:12 +00:00
fi
continue
fi
2023-02-09 12:00:29 +00:00
### Checking if "com.docker.compose.project.config_files" returns the full path to the config file or just the file name
if [ [ $ContConfigFile = '/' * ] ] ; then
ComposeFile = " $ContConfigFile "
else
ComposeFile = " $ContPath / $ContConfigFile "
fi
2023-02-12 18:40:42 +00:00
### cd to the compose-file directory to account for people who use relative volumes, eg - ${PWD}/data:data
2023-03-01 19:52:17 +00:00
cd " $ContPath " || { echo " Path error - skipping $i " ; continue ; }
2023-06-27 17:34:20 +00:00
printf "\n\033[0;36mNow updating (%s/%s): \033[0;34m%s\033[0m\n" " $CurrentQue " " $NumberofUpdates " " $i "
2023-02-23 12:05:06 +00:00
docker pull " $ContImage "
2023-02-25 09:32:22 +00:00
### Reformat for multi-compose:
2023-03-04 18:56:26 +00:00
IFS = ',' read -r -a Confs <<< " $ComposeFile " ; unset IFS
2023-02-25 09:32:22 +00:00
for conf in " ${ Confs [@] } " ; do CompleteConfs += " -f $conf " ; done
2023-02-19 13:21:22 +00:00
### Check if the container got an environment file set, use it if so:
if [ -n " $ContEnv " ] ; then
2023-02-26 12:16:54 +00:00
$DockerBin ${ CompleteConfs [@] } --env-file " $ContEnv " up -d " $ContName " # unquoted array to allow split - rework?
2023-02-19 13:21:22 +00:00
else
2023-02-26 12:16:54 +00:00
$DockerBin ${ CompleteConfs [@] } up -d " $ContName " # unquoted array to allow split - rework?
2023-02-19 13:21:22 +00:00
fi
2023-01-30 09:08:13 +00:00
done
2023-02-15 12:16:31 +00:00
printf "\033[0;32mAll done!\033[0m\n"
[ [ -z " $PruneQ " ] ] && read -r -p "Would you like to prune dangling images? y/[n]: " PruneQ
[ [ " $PruneQ " = ~ [ yY] ] ] && docker image prune -f
2023-01-30 09:08:13 +00:00
else
printf "\nNo updates installed, exiting.\n"
2023-01-20 10:52:47 +00:00
fi
2023-01-18 10:50:00 +00:00
else
2023-01-18 20:45:31 +00:00
printf "\nNo updates available, exiting.\n"
2023-01-18 10:50:00 +00:00
fi
2023-01-30 09:08:13 +00:00
exit 0