| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | #!/usr/bin/env bash
 | 
					
						
							| 
									
										
										
										
											2023-02-07 12:15:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-01 12:13:05 -05:00
										 |  |  | # Copyright (c) 2021-2024 tteck | 
					
						
							| 
									
										
										
										
											2023-02-07 12:15:22 -05:00
										 |  |  | # Author: tteck (tteckster) | 
					
						
							|  |  |  | # License: MIT | 
					
						
							|  |  |  | # https://github.com/tteck/Proxmox/raw/main/LICENSE | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # This sets verbose mode if the global variable is set to "yes" | 
					
						
							| 
									
										
										
										
											2024-01-09 19:44:38 -05:00
										 |  |  | # if [ "$VERBOSE" == "yes" ]; then set -x; fi | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | # This function sets color variables for formatting output in the terminal | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  | YW=$(echo "\033[33m") | 
					
						
							|  |  |  | BL=$(echo "\033[36m") | 
					
						
							|  |  |  | RD=$(echo "\033[01;31m") | 
					
						
							|  |  |  | GN=$(echo "\033[1;92m") | 
					
						
							|  |  |  | CL=$(echo "\033[m") | 
					
						
							| 
									
										
										
										
											2022-04-16 07:21:54 -04:00
										 |  |  | CM="${GN}✓${CL}" | 
					
						
							| 
									
										
										
										
											2022-10-25 03:43:46 -04:00
										 |  |  | CROSS="${RD}✗${CL}" | 
					
						
							| 
									
										
										
										
											2022-04-16 09:52:57 -04:00
										 |  |  | BFR="\\r\\033[K" | 
					
						
							| 
									
										
										
										
											2024-04-21 09:12:55 -04:00
										 |  |  | HOLD=" " | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | # This sets error handling options and defines the error_handler function to handle errors | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  | set -Eeuo pipefail | 
					
						
							|  |  |  | trap 'error_handler $LINENO "$BASH_COMMAND"' ERR | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | # This function handles errors | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  | function error_handler() { | 
					
						
							| 
									
										
										
										
											2024-01-21 04:31:14 -05:00
										 |  |  |   if [ -n "$SPINNER_PID" ] && ps -p $SPINNER_PID > /dev/null; then kill $SPINNER_PID > /dev/null; fi | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  |   printf "\e[?25h" | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  |   local exit_code="$?" | 
					
						
							|  |  |  |   local line_number="$1" | 
					
						
							|  |  |  |   local command="$2" | 
					
						
							|  |  |  |   local error_message="${RD}[ERROR]${CL} in line ${RD}$line_number${CL}: exit code ${RD}$exit_code${CL}: while executing command ${YW}$command${CL}" | 
					
						
							| 
									
										
										
										
											2023-05-06 17:19:12 -04:00
										 |  |  |   echo -e "\n$error_message\n" | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  | # This function displays a spinner. | 
					
						
							|  |  |  | function spinner() { | 
					
						
							| 
									
										
										
										
											2024-04-21 09:12:55 -04:00
										 |  |  |     local chars="/-\|" | 
					
						
							|  |  |  |     local spin_i=0 | 
					
						
							|  |  |  |     printf "\e[?25l" | 
					
						
							|  |  |  |     while true; do | 
					
						
							|  |  |  |         printf "\r \e[36m%s\e[0m" "${chars:spin_i++%${#chars}:1}" | 
					
						
							|  |  |  |         sleep 0.1 | 
					
						
							|  |  |  |     done | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # This function displays an informational message with a yellow color. | 
					
						
							| 
									
										
										
										
											2022-04-25 10:20:22 -04:00
										 |  |  | function msg_info() { | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  |   local msg="$1" | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  |   echo -ne " ${HOLD} ${YW}${msg}   " | 
					
						
							|  |  |  |   spinner & | 
					
						
							|  |  |  |   SPINNER_PID=$! | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-04-25 10:20:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  | # This function displays a success message with a green color. | 
					
						
							| 
									
										
										
										
											2022-04-25 10:20:22 -04:00
										 |  |  | function msg_ok() { | 
					
						
							| 
									
										
										
										
											2024-01-21 04:31:14 -05:00
										 |  |  |   if [ -n "$SPINNER_PID" ] && ps -p $SPINNER_PID > /dev/null; then kill $SPINNER_PID > /dev/null; fi | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  |   printf "\e[?25h" | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  |   local msg="$1" | 
					
						
							|  |  |  |   echo -e "${BFR} ${CM} ${GN}${msg}${CL}" | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  | # This function displays a error message with a red color. | 
					
						
							| 
									
										
										
										
											2022-10-25 03:43:46 -04:00
										 |  |  | function msg_error() { | 
					
						
							| 
									
										
										
										
											2024-01-21 04:31:14 -05:00
										 |  |  |   if [ -n "$SPINNER_PID" ] && ps -p $SPINNER_PID > /dev/null; then kill $SPINNER_PID > /dev/null; fi | 
					
						
							| 
									
										
										
										
											2024-01-06 17:15:45 -05:00
										 |  |  |   printf "\e[?25h" | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  |   local msg="$1" | 
					
						
							|  |  |  |   echo -e "${BFR} ${CROSS} ${RD}${msg}${CL}" | 
					
						
							| 
									
										
										
										
											2022-10-25 03:43:46 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # This checks for the presence of valid Container Storage and Template Storage locations | 
					
						
							| 
									
										
										
										
											2022-10-25 03:43:46 -04:00
										 |  |  | msg_info "Validating Storage" | 
					
						
							| 
									
										
										
										
											2022-10-25 04:43:15 -04:00
										 |  |  | VALIDCT=$(pvesm status -content rootdir | awk 'NR>1') | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  | if [ -z "$VALIDCT" ]; then | 
					
						
							|  |  |  |   msg_error "Unable to detect a valid Container Storage location." | 
					
						
							|  |  |  |   exit 1 | 
					
						
							|  |  |  | fi | 
					
						
							| 
									
										
										
										
											2022-10-25 04:43:15 -04:00
										 |  |  | VALIDTMP=$(pvesm status -content vztmpl | awk 'NR>1') | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  | if [ -z "$VALIDTMP" ]; then | 
					
						
							|  |  |  |   msg_error "Unable to detect a valid Template Storage location." | 
					
						
							|  |  |  |   exit 1 | 
					
						
							|  |  |  | fi | 
					
						
							| 
									
										
										
										
											2022-10-25 04:43:15 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # This function is used to select the storage class and determine the corresponding storage content type and label. | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | function select_storage() { | 
					
						
							|  |  |  |   local CLASS=$1 | 
					
						
							|  |  |  |   local CONTENT | 
					
						
							|  |  |  |   local CONTENT_LABEL | 
					
						
							|  |  |  |   case $CLASS in | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  |   container) | 
					
						
							|  |  |  |     CONTENT='rootdir' | 
					
						
							|  |  |  |     CONTENT_LABEL='Container' | 
					
						
							|  |  |  |     ;; | 
					
						
							|  |  |  |   template) | 
					
						
							|  |  |  |     CONTENT='vztmpl' | 
					
						
							|  |  |  |     CONTENT_LABEL='Container template' | 
					
						
							|  |  |  |     ;; | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  |   *) false || exit "Invalid storage class." ;; | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  |   esac | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  |    | 
					
						
							|  |  |  |   # This Queries all storage locations | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  |   local -a MENU | 
					
						
							|  |  |  |   while read -r line; do | 
					
						
							|  |  |  |     local TAG=$(echo $line | awk '{print $1}') | 
					
						
							|  |  |  |     local TYPE=$(echo $line | awk '{printf "%-10s", $2}') | 
					
						
							|  |  |  |     local FREE=$(echo $line | numfmt --field 4-6 --from-unit=K --to=iec --format %.2f | awk '{printf( "%9sB", $6)}') | 
					
						
							|  |  |  |     local ITEM="  Type: $TYPE Free: $FREE " | 
					
						
							|  |  |  |     local OFFSET=2 | 
					
						
							|  |  |  |     if [[ $((${#ITEM} + $OFFSET)) -gt ${MSG_MAX_LENGTH:-} ]]; then | 
					
						
							|  |  |  |       local MSG_MAX_LENGTH=$((${#ITEM} + $OFFSET)) | 
					
						
							|  |  |  |     fi | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  |     MENU+=("$TAG" "$ITEM" "OFF") | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  |   done < <(pvesm status -content $CONTENT | awk 'NR>1') | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  |    | 
					
						
							|  |  |  |   # Select storage location | 
					
						
							| 
									
										
										
										
											2023-08-19 06:22:12 -04:00
										 |  |  |   if [ $((${#MENU[@]}/3)) -eq 1 ]; then | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  |     printf ${MENU[0]} | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  |     local STORAGE | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  |     while [ -z "${STORAGE:+x}" ]; do | 
					
						
							| 
									
										
										
										
											2023-09-09 05:13:17 -04:00
										 |  |  |       STORAGE=$(whiptail --backtitle "Proxmox VE Helper Scripts" --title "Storage Pools" --radiolist \
 | 
					
						
							| 
									
										
										
										
											2023-08-19 06:22:12 -04:00
										 |  |  |       "Which storage pool you would like to use for the ${CONTENT_LABEL,,}?\nTo make a selection, use the Spacebar.\n" \
 | 
					
						
							|  |  |  |       16 $(($MSG_MAX_LENGTH + 23)) 6 \
 | 
					
						
							|  |  |  |       "${MENU[@]}" 3>&1 1>&2 2>&3) || exit "Menu aborted." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  |     done | 
					
						
							|  |  |  |     printf $STORAGE | 
					
						
							|  |  |  |   fi | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Test if required variables are set | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  | [[ "${CTID:-}" ]] || exit "You need to set 'CTID' variable." | 
					
						
							|  |  |  | [[ "${PCT_OSTYPE:-}" ]] || exit "You need to set 'PCT_OSTYPE' variable." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Test if ID is valid | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  | [ "$CTID" -ge "100" ] || exit "ID cannot be less than 100." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Test if ID is in use | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | if pct status $CTID &>/dev/null; then | 
					
						
							| 
									
										
										
										
											2022-05-21 07:11:10 -04:00
										 |  |  |   echo -e "ID '$CTID' is already in use." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  |   unset CTID | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  |   exit "Cannot use ID that is already in use." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | fi | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Get template storage | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | TEMPLATE_STORAGE=$(select_storage template) || exit | 
					
						
							| 
									
										
										
										
											2022-04-25 10:27:30 -04:00
										 |  |  | msg_ok "Using ${BL}$TEMPLATE_STORAGE${CL} ${GN}for Template Storage." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Get container storage | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | CONTAINER_STORAGE=$(select_storage container) || exit | 
					
						
							| 
									
										
										
										
											2022-04-25 10:27:30 -04:00
										 |  |  | msg_ok "Using ${BL}$CONTAINER_STORAGE${CL} ${GN}for Container Storage." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Update LXC template list | 
					
						
							| 
									
										
										
										
											2022-04-16 09:52:57 -04:00
										 |  |  | msg_info "Updating LXC Template List" | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | pveam update >/dev/null | 
					
						
							| 
									
										
										
										
											2022-04-16 11:57:11 -04:00
										 |  |  | msg_ok "Updated LXC Template List" | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Get LXC template string | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | TEMPLATE_SEARCH=${PCT_OSTYPE}-${PCT_OSVERSION:-} | 
					
						
							|  |  |  | mapfile -t TEMPLATES < <(pveam available -section system | sed -n "s/.*\($TEMPLATE_SEARCH.*\)/\1/p" | sort -t - -k 2 -V) | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  | [ ${#TEMPLATES[@]} -gt 0 ] || exit "Unable to find a template when searching for '$TEMPLATE_SEARCH'." | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | TEMPLATE="${TEMPLATES[-1]}" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Download LXC template if needed | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | if ! pveam list $TEMPLATE_STORAGE | grep -q $TEMPLATE; then | 
					
						
							| 
									
										
										
										
											2022-04-16 09:52:57 -04:00
										 |  |  |   msg_info "Downloading LXC Template" | 
					
						
							| 
									
										
										
										
											2023-02-03 10:53:08 -05:00
										 |  |  |   pveam download $TEMPLATE_STORAGE $TEMPLATE >/dev/null || | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  |     exit "A problem occured while downloading the LXC template." | 
					
						
							| 
									
										
										
										
											2022-04-20 14:27:25 -04:00
										 |  |  |   msg_ok "Downloaded LXC Template" | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | fi | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Combine all options | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | DEFAULT_PCT_OPTIONS=( | 
					
						
							|  |  |  |   -arch $(dpkg --print-architecture)) | 
					
						
							| 
									
										
										
										
											2022-10-29 20:08:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | PCT_OPTIONS=(${PCT_OPTIONS[@]:-${DEFAULT_PCT_OPTIONS[@]}}) | 
					
						
							|  |  |  | [[ " ${PCT_OPTIONS[@]} " =~ " -rootfs " ]] || PCT_OPTIONS+=(-rootfs $CONTAINER_STORAGE:${PCT_DISK_SIZE:-8}) | 
					
						
							| 
									
										
										
										
											2022-02-23 06:52:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 19:15:26 -04:00
										 |  |  | # Create container | 
					
						
							| 
									
										
										
										
											2022-04-16 10:05:03 -04:00
										 |  |  | msg_info "Creating LXC Container" | 
					
						
							| 
									
										
										
										
											2023-02-03 10:53:08 -05:00
										 |  |  | pct create $CTID ${TEMPLATE_STORAGE}:vztmpl/${TEMPLATE} ${PCT_OPTIONS[@]} >/dev/null || | 
					
						
							| 
									
										
										
										
											2023-02-03 11:41:12 -05:00
										 |  |  |   exit "A problem occured while trying to create container." | 
					
						
							| 
									
										
										
										
											2022-04-16 11:46:18 -04:00
										 |  |  | msg_ok "LXC Container ${BL}$CTID${CL} ${GN}was successfully created." |