2022-04-21 21:19:53 +00:00
#!/bin/bash
#If no argument is passed, kill the script.
2022-04-22 21:37:02 +00:00
[ [ -z " $* " ] ] && echo "This script requires an arguent, use -h for help" && exit
2022-04-21 21:19:53 +00:00
2022-05-10 03:54:31 +00:00
while getopts ":hsi:mrb:t:uUpSRv" opt
2022-04-21 21:19:53 +00:00
do
2022-05-10 03:54:31 +00:00
case $opt in
h)
echo "-m | Initiates mounting feature, choose between unmounting and mounting PVC data"
echo "-r | Opens a menu to restore a heavy_script backup that was taken on you ix-applications pool"
echo "-b | Back-up your ix-applications dataset, specify a number after -b"
echo "-i | Add application to ignore list, one by one, see example below."
echo "-R | Roll-back applications if they fail to update"
echo "-S | Shutdown applications prior to updating"
echo "-v | verbose output"
echo "-t | Set a custom timeout in seconds when checking if either an App or Mountpoint correctly Started, Stopped or (un)Mounted. Defaults to 500 seconds"
echo "-s | sync catalog"
echo "-S | Stops App before update with -u or -U and restarts afterwards"
echo "-U | Update all applications, ignores versions"
echo "-u | Update all applications, does not update Major releases"
echo "-p | Prune unused/old docker images"
echo "-s | Stop App before attempting update"
echo "EX | bash heavy_script.sh -b 14 -i portainer -i arch -i sonarr -i radarr -t 600 -vRsUp"
echo "EX | bash /mnt/tank/scripts/heavy_script.sh -t 8812 -m"
exit
; ;
\? )
echo " Invalid Option - $OPTARG , type -h for help "
exit
; ;
:)
echo " Option: - $OPTARG requires an argument " >& 2
exit
; ;
b)
re = '^[0-9]+$'
number_of_backups = $OPTARG
2022-05-12 04:28:27 +00:00
! [ [ $OPTARG = ~ $re ] ] && echo -e "Error: -b needs to be assigned an interger\n" $number_of_backups " is not an interger" >& 2 && exit
[ [ " $number_of_backups " -le 0 ] ] && echo "Error: Number of backups is required to be at least 1" && exit
2022-05-10 03:54:31 +00:00
; ;
r)
restore = "true"
; ;
i)
ignore += ( " $OPTARG " )
; ;
t)
re = '^[0-9]+$'
timeout = $OPTARG
2022-05-12 04:28:27 +00:00
! [ [ $timeout = ~ $re ] ] && echo -e "Error: -t needs to be assigned an interger\n" $timeout " is not an interger" >& 2 && exit
[ [ " $timeout " -le 50 ] ] && echo "Warning: Your timeout is set very low and may lead to premature rollbacks or skips"
2022-05-10 03:54:31 +00:00
; ;
m)
mount = "true"
; ;
s)
sync = "true"
; ;
U)
update_all_apps = "true"
; ;
u)
update_apps = "true"
; ;
S)
stop_before_update = "true"
; ;
p)
prune = "true"
; ;
R)
rollback = "true"
; ;
v)
verbose = "true"
; ;
esac
done
backup( ) {
2022-05-12 04:28:27 +00:00
echo -e " \nNumber of backups was set to $number_of_backups "
2022-05-10 03:54:31 +00:00
date = $( date '+%Y_%m_%d_%H_%M_%S' )
[ [ " $verbose " = = "true" ] ] && cli -c 'app kubernetes backup_chart_releases backup_name=' '"' HeavyScript_" $date " '"'
[ [ -z " $verbose " ] ] && echo -e "\nNew Backup Name:" && cli -c 'app kubernetes backup_chart_releases backup_name=' '"' HeavyScript_" $date " '"' | tail -n 1
mapfile -t list_backups < <( cli -c 'app kubernetes list_backups' | grep "HeavyScript_" | sort -t '_' -Vr -k2,7 | awk -F '|' '{print $2}' | tr -d " \t\r" )
if [ [ ${# list_backups [@] } -gt "number_of_backups" ] ] ; then
echo -e "\nDeleting the oldest backup(s) for exceeding limit:"
overflow = $( expr ${# list_backups [@] } - $number_of_backups )
2022-05-12 14:54:48 +00:00
mapfile -t list_overflow < <( cli -c 'app kubernetes list_backups' | grep "HeavyScript_" | sort -t '_' -V -k2,7 | awk -F '|' '{print $2}' | tr -d " \t\r" | head -n " $overflow " )
2022-05-10 03:54:31 +00:00
for i in " ${ list_overflow [@] } "
do
cli -c 'app kubernetes delete_backup backup_name=' '"' " $i " '"' & > /dev/null || echo " Failed to delete $i "
echo " $i "
done
fi
}
export -f backup
restore( ) {
2022-05-11 02:53:01 +00:00
clear -x
2022-05-12 04:28:27 +00:00
list_backups = $( cli -c 'app kubernetes list_backups' | grep "HeavyScript_" | sort -t '_' -Vr -k2,7 | tr -d " \t\r" | awk -F '|' '{print $2}' | nl | column -t)
2022-05-10 23:56:23 +00:00
echo " $list_backups " && read -p "Please type a number: " selection && restore_point = $( echo " $list_backups " | grep ^" $selection " | awk '{print $2}' )
2022-05-11 02:53:01 +00:00
[ [ -z " $selection " ] ] && echo "Your selection cannot be empty" && exit #Check for valid selection. If none, kill script
2022-05-10 23:56:23 +00:00
[ [ -z " $restore_point " ] ] && echo " Invalid Selection: $selection , was not an option " && exit #Check for valid selection. If none, kill script
2022-05-11 02:53:01 +00:00
echo -e "\nWARNING:\nThis is NOT guranteed to work\nThis is ONLY supposed to be used as a LAST RESORT\nConsider rolling back your applications instead if possible" || { echo "FAILED" ; exit; }
echo -e " \n\nYou have chosen:\n $restore_point \n\nWould you like to continue? " && echo -e "1 Yes\n2 No" && read -p "Please type a number: " yesno || { echo "FAILED" ; exit; }
2022-05-10 03:54:31 +00:00
if [ [ $yesno = = "1" ] ] ; then
echo -e "\nStarting Backup, this will take a LONG time." && cli -c 'app kubernetes restore_backup backup_name=' '"' " $restore_point " '"' || echo "Restore FAILED"
elif [ [ $yesno = = "2" ] ] ; then
echo "You've chosen NO, killing script. Good luck."
else
echo "Invalid Selection"
fi
}
export -f restore
mount( ) {
2022-05-11 02:53:01 +00:00
clear -x
2022-05-10 03:54:31 +00:00
echo -e "1 Mount\n2 Unmount All" && read -p "Please type a number: " selection
2022-05-11 02:53:01 +00:00
[ [ -z " $selection " ] ] && echo "Your selection cannot be empty" && exit #Check for valid selection. If none, kill script
2022-05-10 03:54:31 +00:00
if [ [ $selection = = "1" ] ] ; then
list = $( k3s kubectl get pvc -A | sort -u | awk '{print NR-1, "\t" $1 "\t" $2 "\t" $4}' | column -t | sed "s/^0/ /" )
echo " $list " && read -p "Please type a number : " selection
2022-05-11 02:53:01 +00:00
[ [ -z " $selection " ] ] && echo "Your selection cannot be empty" && exit #Check for valid selection. If none, kill script
2022-05-10 23:56:23 +00:00
app = $( echo -e " $list " | grep ^" $selection " | awk '{print $2}' | cut -c 4- )
[ [ -z " $app " ] ] && echo " Invalid Selection: $selection , was not an option " && exit #Check for valid selection. If none, kill script
pvc = $( echo -e " $list " | grep ^" $selection " )
2022-05-10 03:54:31 +00:00
status = $( cli -m csv -c 'app chart_release query name,status' | grep -E " (,|^) $app (,| $) " | awk -F ',' '{print $2}' | tr -d " \t\n\r" )
if [ [ " $status " != "STOPPED" ] ] ; then
2022-05-10 18:48:35 +00:00
[ [ -z $timeout ] ] && echo -e "\nDefault Timeout: 500" && timeout = 500 || echo -e " \nCustom Timeout: $timeout "
2022-05-10 03:54:31 +00:00
SECONDS = 0 && echo -e " \nScaling down $app " && midclt call chart.release.scale " $app " '{"replica_count": 0}' & > /dev/null
else
echo -e " \n $app is already stopped "
fi
while [ [ " $SECONDS " -le " $timeout " && " $status " != "STOPPED" ] ]
do
status = $( cli -m csv -c 'app chart_release query name,status' | grep -E " (,|^) $app (,| $) " | awk -F ',' '{print $2}' | tr -d " \t\n\r" )
echo -e " Waiting $(( timeout-SECONDS)) more seconds for $app to be STOPPED " && sleep 10
done
data_name = $( echo " $pvc " | awk '{print $3}' )
mount = $( echo " $pvc " | awk '{print $4}' )
volume_name = $( echo " $pvc " | awk '{print $4}' )
full_path = $( zfs list | grep $volume_name | awk '{print $1}' )
2022-05-12 04:28:27 +00:00
echo -e "\nMounting\n" $full_path " \nTo\n/mnt/heavyscript/ $data_name " && zfs set mountpoint = /heavyscript/" $data_name " " $full_path " && echo -e "Mounted\n\nUnmount with the following command\nzfs set mountpoint=legacy " $full_path " && rmdir /mnt/heavyscript/" $data_name "\nOr use the Unmount All option\n"
2022-05-10 03:54:31 +00:00
exit
elif [ [ $selection = = "2" ] ] ; then
2022-05-12 04:28:27 +00:00
mapfile -t unmount_array < <( basename -a /mnt/heavyscript/* | sed "s/*//" )
2022-05-10 03:54:31 +00:00
[ [ -z $unmount_array ] ] && echo "Theres nothing to unmount" && exit
for i in " ${ unmount_array [@] } "
do
main = $( k3s kubectl get pvc -A | grep " $i " | awk '{print $1, $2, $4}' )
app = $( echo " $main " | awk '{print $1}' | cut -c 4-)
pvc = $( echo " $main " | awk '{print $3}' )
2022-05-11 02:53:01 +00:00
path = $( find /mnt/*/ix-applications/releases/" $app " /volumes/ -maxdepth 0 | cut -c 6-)
2022-05-12 04:28:27 +00:00
safety_check = $( find /mnt/*/ix-applications/releases/" $app " /volumes/ -maxdepth 0 | cut -c 6- | wc -l) #if theres more than one new lines, that means theres more than one application with the same name on another pool.
if [ [ " $safety_check " -gt 1 ] ] ; then #if there is another app with the same name on another pool, use the current pools application, since the other instance is probably old, or unused.
echo " $i is a name used on more than one pool.. attempting to use your current kubernetes apps pool "
pool = $( cli -c 'app kubernetes config' | grep dataset | awk -F '|' '{print $3}' | awk -F '/' '{print $1}' | tr -d " \t\n\r" )
full_path = $( find /mnt/" $pool " /ix-applications/releases/" $app " /volumes/ -maxdepth 0 | cut -c 6-)
zfs set mountpoint = legacy " $full_path " " $pvc " && echo " $i unmounted " && rmdir /mnt/heavyscript/" $i " && continue || echo " failed to unmount $i "
fi
zfs set mountpoint = legacy " $path " " $pvc " && echo " $i unmounted " && rmdir /mnt/heavyscript/" $i " || echo " failed to unmount $i "
2022-05-10 03:54:31 +00:00
done
2022-05-12 04:28:27 +00:00
rmdir /mnt/heavyscript
2022-05-10 03:54:31 +00:00
else
echo "Invalid selection, type -h for help"
fi
}
export -f mount
sync( ) {
echo -e "\nSyncing all catalogs, please wait.." && cli -c 'app catalog sync_all' & > /dev/null && echo -e "Catalog sync complete"
}
export -f sync
prune( ) {
echo -e "\nPruning Docker Images" && docker image prune -af | grep Total || echo "Failed to Prune Docker Images"
}
export -f prune
update_apps( ) {
mapfile -t array < <( cli -m csv -c 'app chart_release query name,update_available,human_version,human_latest_version,container_images_update_available,status' | grep ",true" | sort)
2022-05-13 16:46:15 +00:00
[ [ -z $array ] ] && echo -e "\nThere are no updates available" && return 0 || echo -e " \n ${# array [@] } update(s) available "
2022-05-10 18:48:35 +00:00
[ [ -z $timeout ] ] && echo -e "\nDefault Timeout: 500" && timeout = 500 || echo -e " \nCustom Timeout: $timeout "
2022-05-10 03:54:31 +00:00
for i in " ${ array [@] } "
2022-04-28 00:37:36 +00:00
do
2022-05-10 03:54:31 +00:00
app_name = $( echo " $i " | awk -F ',' '{print $1}' ) #print out first catagory, name.
old_app_ver = $( echo " $i " | awk -F ',' '{print $4}' | awk -F '_' '{print $1}' | awk -F '.' '{print $1}' ) #previous/current Application MAJOR Version
new_app_ver = $( echo " $i " | awk -F ',' '{print $5}' | awk -F '_' '{print $1}' | awk -F '.' '{print $1}' ) #new Application MAJOR Version
old_chart_ver = $( echo " $i " | awk -F ',' '{print $4}' | awk -F '_' '{print $2}' | awk -F '.' '{print $1}' ) # Old Chart MAJOR version
new_chart_ver = $( echo " $i " | awk -F ',' '{print $5}' | awk -F '_' '{print $2}' | awk -F '.' '{print $1}' ) # New Chart MAJOR version
status = $( echo " $i " | awk -F ',' '{print $2}' ) #status of the app: STOPPED / DEPLOYING / ACTIVE
diff_app = $( diff <( echo " $old_app_ver " ) <( echo " $new_app_ver " ) ) #caluclating difference in major app versions
diff_chart = $( diff <( echo " $old_chart_ver " ) <( echo " $new_chart_ver " ) ) #caluclating difference in Chart versions
old_full_ver = $( echo " $i " | awk -F ',' '{print $4}' ) #Upgraded From
new_full_ver = $( echo " $i " | awk -F ',' '{print $5}' ) #Upraded To
rollback_version = $( echo " $i " | awk -F ',' '{print $4}' | awk -F '_' '{print $2}' )
2022-05-10 20:46:47 +00:00
printf '%s\0' " ${ ignore [@] } " | grep -iFxqz " ${ app_name } " && echo -e " \n $app_name \nIgnored, skipping " && continue #If application is on ignore list, skip
2022-05-10 03:54:31 +00:00
if [ [ " $diff_app " = = " $diff_chart " || " $update_all_apps " = = "true" ] ] ; then #continue to update
startstatus = $status
if [ [ $stop_before_update = = "true" ] ] ; then # Check to see if user is using -S or not
if [ [ " $status " = = "STOPPED" ] ] ; then # if status is already stopped, skip while loop
echo -e " \n $app_name "
2022-05-11 01:35:46 +00:00
[ [ " $verbose " = = "true" ] ] && echo "Updating.."
2022-05-10 03:54:31 +00:00
cli -c 'app chart_release upgrade release_name=' '"' " $app_name " '"' & > /dev/null && echo -e " Updated\n $old_full_ver \n $new_full_ver " && after_update_actions || echo "FAILED"
continue
else # if status was not STOPPED, stop the app prior to updating
echo -e "\n" $app_name ""
2022-05-11 01:35:46 +00:00
[ [ " $verbose " = = "true" ] ] && echo "Stopping prior to update.."
2022-05-10 03:54:31 +00:00
midclt call chart.release.scale " $app_name " '{"replica_count": 0}' & > /dev/null && SECONDS = 0 || echo -e "FAILED"
while [ [ " $status " != "STOPPED" ] ]
do
status = $( cli -m csv -c 'app chart_release query name,update_available,human_version,human_latest_version,status' | grep "" $app_name "," | awk -F ',' '{print $2}' )
if [ [ " $status " = = "STOPPED" ] ] ; then
echo "Stopped"
2022-05-11 01:35:46 +00:00
[ [ " $verbose " = = "true" ] ] && echo "Updating.."
2022-05-10 03:54:31 +00:00
cli -c 'app chart_release upgrade release_name=' '"' " $app_name " '"' & > /dev/null && echo -e " Updated\n $old_full_ver \n $new_full_ver " && after_update_actions || echo "Failed to update"
break
elif [ [ " $SECONDS " -ge " $timeout " ] ] ; then
echo " Error: Run Time( $SECONDS ) has exceeded Timeout( $timeout ) "
break
elif [ [ " $status " != "STOPPED" ] ] ; then
[ [ " $verbose " = = "true" ] ] && echo " Waiting $(( timeout-SECONDS)) more seconds for $app_name to be STOPPED "
sleep 10
continue
fi
done
fi
else #user must not be using -S, just update
echo -e " \n $app_name "
2022-05-11 01:35:46 +00:00
[ [ " $verbose " = = "true" ] ] && echo "Updating.."
2022-05-10 03:54:31 +00:00
cli -c 'app chart_release upgrade release_name=' '"' " $app_name " '"' & > /dev/null && echo -e " Updated\n $old_full_ver \n $new_full_ver " && after_update_actions || { echo "FAILED" ; continue ; }
fi
else
echo -e " \n $app_name \nMajor Release, update manually "
continue
fi
2022-04-28 00:37:36 +00:00
done
2022-05-10 03:54:31 +00:00
}
export -f update_apps
2022-04-21 21:19:53 +00:00
2022-05-10 03:54:31 +00:00
after_update_actions( ) {
SECONDS = 0
count = 0
if [ [ $rollback = = "true" ] ] ; then
while [ [ "0" != "1" ] ]
do
( ( count++ ) )
status = $( cli -m csv -c 'app chart_release query name,update_available,human_version,human_latest_version,status' | grep "" $app_name "," | awk -F ',' '{print $2}' )
if [ [ " $status " = = "ACTIVE" && " $startstatus " = = "STOPPED" ] ] ; then
[ [ " $verbose " = = "true" ] ] && echo "Returing to STOPPED state.."
midclt call chart.release.scale " $app_name " '{"replica_count": 0}' & > /dev/null && echo "Stopped" || echo "FAILED"
break
elif [ [ " $SECONDS " -ge " $timeout " && " $status " = = "DEPLOYING" && " $failed " != "true" ] ] ; then
2022-05-11 01:35:46 +00:00
echo -e " Error: Run Time( $SECONDS ) for $app_name has exceeded Timeout( $timeout )\nIf this is a slow starting application, set a higher timeout with -t\nIf this applicaion is always DEPLOYING, you can disable all probes under the Healthcheck Probes Liveness section in the edit configuration\nReverting update.. "
2022-05-10 03:54:31 +00:00
midclt call chart.release.rollback " $app_name " " {\"item_version\": \" $rollback_version \"} " & > /dev/null
[ [ " $startstatus " = = "STOPPED" ] ] && failed = "true" && after_update_actions && unset failed #run back after_update_actions function if the app was stopped prior to update
break
elif [ [ " $SECONDS " -ge " $timeout " && " $status " = = "DEPLOYING" && " $failed " = = "true" ] ] ; then
echo -e " Error: Run Time( $SECONDS ) for $app_name has exceeded Timeout( $timeout )\nThe application failed to be ACTIVE even after a rollback,\nManual intervention is required\nAbandoning "
break
elif [ [ " $status " = = "STOPPED" ] ] ; then
2022-05-11 01:35:46 +00:00
[ [ " $count " -le 1 && " $verbose " = = "true" ] ] && echo "Verifying Stopped.." && sleep 15 && continue #if reports stopped on FIRST time through loop, double check
[ [ " $count " -le 1 && -z " $verbose " ] ] && sleep 15 && continue #if reports stopped on FIRST time through loop, double check
echo "Stopped" && break #if reports stopped any time after the first loop, assume its extermal services.
2022-05-10 03:54:31 +00:00
elif [ [ " $status " = = "ACTIVE" ] ] ; then
2022-05-11 01:35:46 +00:00
[ [ " $count " -le 1 && " $verbose " = = "true" ] ] && echo "Verifying Active.." && sleep 15 && continue #if reports active on FIRST time through loop, double check
[ [ " $count " -le 1 && -z " $verbose " ] ] && sleep 15 && continue #if reports active on FIRST time through loop, double check
2022-05-10 23:37:20 +00:00
echo "Active" && break #if reports active any time after the first loop, assume actually active.
2022-05-10 03:54:31 +00:00
else
[ [ " $verbose " = = "true" ] ] && echo " Waiting $(( timeout-SECONDS)) more seconds for $app_name to be ACTIVE "
sleep 15
continue
fi
done
else
if [ [ " $startstatus " = = "STOPPED" ] ] ; then
2022-05-10 22:42:05 +00:00
while [ [ "0" != "1" ] ]
2022-05-10 03:54:31 +00:00
do
( ( count++ ) )
2022-05-12 04:28:27 +00:00
status = $( cli -m csv -c 'app chart_release query name,update_available,human_version,human_latest_version,status' | grep "" $app_name "," | awk -F ',' '{print $2}' )
2022-05-10 03:54:31 +00:00
if [ [ " $status " = = "STOPPED" ] ] ; then
2022-05-11 01:35:46 +00:00
[ [ " $count " -le 1 && " $verbose " = = "true" ] ] && echo "Verifying Stopped.." && sleep 15 && continue #if reports stopped on FIRST time through loop, double check
[ [ " $count " -le 1 && -z " $verbose " ] ] && sleep 15 && continue #if reports stopped on FIRST time through loop, double check
2022-05-10 23:37:20 +00:00
echo "Stopped" && break #assume actually stopped anytime AFTER the first loop
2022-04-21 21:19:53 +00:00
break
2022-05-10 03:54:31 +00:00
elif [ [ " $status " = = "ACTIVE" ] ] ; then
2022-05-11 01:35:46 +00:00
[ [ " $count " -le 1 && " $verbose " = = "true" ] ] && echo "Verifying Active.." && sleep 15 && continue #if reports active on FIRST time through loop, double check
[ [ " $count " -le 1 && -z " $verbose " ] ] && sleep 15 && continue #if reports active on FIRST time through loop, double check
2022-05-10 03:54:31 +00:00
[ [ " $verbose " = = "true" ] ] && echo "Returing to STOPPED state.."
midclt call chart.release.scale " $app_name " '{"replica_count": 0}' & > /dev/null && echo "Stopped" || echo "FAILED"
2022-04-21 21:19:53 +00:00
break
2022-05-10 03:54:31 +00:00
elif [ [ " $SECONDS " -ge " $timeout " ] ] ; then
echo " Error: Run Time( $SECONDS ) has exceeded Timeout( $timeout ) "
break
2022-05-10 22:42:05 +00:00
else
2022-05-10 03:54:31 +00:00
[ [ " $verbose " = = "true" ] ] && echo " Waiting $(( timeout-SECONDS)) more seconds for $app_name to be ACTIVE "
sleep 10
continue
2022-04-21 21:19:53 +00:00
fi
2022-05-10 03:54:31 +00:00
done
fi
fi
}
export -f prune
2022-05-12 04:28:27 +00:00
[ [ " $restore " = = "true" && " $mount " = = "true" ] ] && echo -e "The Restore Function(-r)\nand\nMount Function(-m)\nCannot both be called at the same time." && exit
[ [ " $restore " = = "true" ] ] && restore && exit
[ [ " $mount " = = "true" ] ] && mount && exit
[ [ " $number_of_backups " -ge 1 ] ] && backup
[ [ " $sync " = = "true" ] ] && sync
[ [ " $update_all_apps " = = "true" || $update_apps = = "true" ] ] && update_apps
[ [ " $prune " = = "true" ] ] && prune