haos_vm.sh 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. #!/usr/bin/env bash
  2. # Setup script environment
  3. set -o errexit #Exit immediately if a pipeline returns a non-zero status
  4. set -o errtrace #Trap ERR from shell functions, command substitutions, and commands from subshell
  5. set -o nounset #Treat unset variables as an error
  6. set -o pipefail #Pipe will exit with last non-zero status if applicable
  7. shopt -s expand_aliases
  8. alias die='EXIT=$? LINE=$LINENO error_exit'
  9. trap die ERR
  10. trap cleanup EXIT
  11. function error_exit() {
  12. trap - ERR
  13. local DEFAULT='Unknown failure occured.'
  14. local REASON="\e[97m${1:-$DEFAULT}\e[39m"
  15. local FLAG="\e[91m[ERROR] \e[93m$EXIT@$LINE"
  16. msg "$FLAG $REASON"
  17. [ ! -z ${VMID-} ] && cleanup_vmid
  18. exit $EXIT
  19. }
  20. function warn() {
  21. local REASON="\e[97m$1\e[39m"
  22. local FLAG="\e[93m[WARNING]\e[39m"
  23. msg "$FLAG $REASON"
  24. }
  25. function info() {
  26. local REASON="$1"
  27. local FLAG="\e[36m[INFO]\e[39m"
  28. msg "$FLAG $REASON"
  29. }
  30. function msg() {
  31. local TEXT="$1"
  32. echo -e "$TEXT"
  33. }
  34. function cleanup_vmid() {
  35. if $(qm status $VMID &>/dev/null); then
  36. if [ "$(qm status $VMID | awk '{print $2}')" == "running" ]; then
  37. qm stop $VMID
  38. fi
  39. qm destroy $VMID
  40. fi
  41. }
  42. function cleanup() {
  43. popd >/dev/null
  44. rm -rf $TEMP_DIR
  45. }
  46. TEMP_DIR=$(mktemp -d)
  47. pushd $TEMP_DIR >/dev/null
  48. # Select storage location
  49. while read -r line; do
  50. TAG=$(echo $line | awk '{print $1}')
  51. TYPE=$(echo $line | awk '{printf "%-10s", $2}')
  52. FREE=$(echo $line | numfmt --field 4-6 --from-unit=K --to=iec --format %.2f | awk '{printf( "%9sB", $6)}')
  53. ITEM=" Type: $TYPE Free: $FREE "
  54. OFFSET=2
  55. if [[ $((${#ITEM} + $OFFSET)) -gt ${MSG_MAX_LENGTH:-} ]]; then
  56. MSG_MAX_LENGTH=$((${#ITEM} + $OFFSET))
  57. fi
  58. STORAGE_MENU+=( "$TAG" "$ITEM" "OFF" )
  59. done < <(pvesm status -content images | awk 'NR>1')
  60. if [ $((${#STORAGE_MENU[@]}/3)) -eq 0 ]; then
  61. warn "'Disk image' needs to be selected for at least one storage location."
  62. die "Unable to detect valid storage location."
  63. elif [ $((${#STORAGE_MENU[@]}/3)) -eq 1 ]; then
  64. STORAGE=${STORAGE_MENU[0]}
  65. else
  66. while [ -z "${STORAGE:+x}" ]; do
  67. STORAGE=$(whiptail --title "Storage Pools" --radiolist \
  68. "Which storage pool you would like to use for the container?\n\n" \
  69. 16 $(($MSG_MAX_LENGTH + 23)) 6 \
  70. "${STORAGE_MENU[@]}" 3>&1 1>&2 2>&3) || exit
  71. done
  72. fi
  73. info "Using '$STORAGE' for storage location."
  74. # Get the next guest VM/LXC ID
  75. VMID=$(pvesh get /cluster/nextid)
  76. info "Container ID is $VMID."
  77. # Get latest Home Assistant disk image archive URL
  78. echo -e "\e[1;33m Getting URL for latest Home Assistant disk image... \e[0m"
  79. RELEASE_TYPE=qcow2
  80. URL=$(cat<<EOF | python3
  81. import requests
  82. url = "https://api.github.com/repos/home-assistant/operating-system/releases"
  83. r = requests.get(url).json()
  84. if "message" in r:
  85. exit()
  86. for release in r:
  87. if release["prerelease"]:
  88. continue
  89. for asset in release["assets"]:
  90. if asset["name"].find("$RELEASE_TYPE") != -1:
  91. image_url = asset["browser_download_url"]
  92. print(image_url)
  93. exit()
  94. EOF
  95. )
  96. if [ -z "$URL" ]; then
  97. die "Github has returned an error. A rate limit may have been applied to your connection."
  98. fi
  99. # Download Home Assistant disk image archive
  100. echo -e "\e[1;33m Downloading disk image... \e[0m"
  101. wget -q --show-progress $URL
  102. echo -en "\e[1A\e[0K" #Overwrite output from wget
  103. FILE=$(basename $URL)
  104. # Check for and Install unzip (if needed)
  105. if [[ $FILE == *.zip ]]; then
  106. echo -e "\e[1;33m Checking for unzip command... \e[0m"
  107. if ! command -v unzip &> /dev/null; then
  108. echo -e "\e[1;33m Installing Unzip... \e[0m"
  109. apt-get --allow-releaseinfo-change update >/dev/null
  110. apt-get -qqy install unzip &>/dev/null
  111. fi
  112. fi
  113. # Extract Home Assistant disk image
  114. echo -e "\e[1;33m Extracting disk image... \e[0m"
  115. case $FILE in
  116. *"gz") gunzip -f $FILE;;
  117. *"zip") unzip -o $FILE;;
  118. *"xz") xz -d $FILE;;
  119. *) die "Unable to handle file extension '${FILE##*.}'.";;
  120. esac
  121. # Create variables for container disk
  122. STORAGE_TYPE=$(pvesm status -storage $STORAGE | awk 'NR>1 {print $2}')
  123. case $STORAGE_TYPE in
  124. nfs|dir)
  125. DISK_EXT=".qcow2"
  126. DISK_REF="$VMID/"
  127. IMPORT_OPT="-format qcow2"
  128. esac
  129. for i in {0,1}; do
  130. disk="DISK$i"
  131. eval DISK${i}=vm-${VMID}-disk-${i}${DISK_EXT:-}
  132. eval DISK${i}_REF=${STORAGE}:${DISK_REF:-}${!disk}
  133. done
  134. # Create VM
  135. echo -e "\e[1;33m Creating VM... \e[0m"
  136. VM_NAME=$(sed -e "s/\_//g" -e "s/.${RELEASE_TYPE}.*$//" <<< $FILE)
  137. qm create $VMID -agent 1 -bios ovmf -cores 2 -memory 4096 -name $VM_NAME -net0 virtio,bridge=vmbr0 \
  138. -onboot 1 -ostype l26 -scsihw virtio-scsi-pci
  139. pvesm alloc $STORAGE $VMID $DISK0 128 1>&/dev/null
  140. qm importdisk $VMID ${FILE%.*} $STORAGE ${IMPORT_OPT:-} 1>&/dev/null
  141. qm set $VMID \
  142. -efidisk0 ${DISK0_REF},size=128K \
  143. -sata0 ${DISK1_REF},size=32G > /dev/null
  144. qm set $VMID \
  145. -boot order=sata0 > /dev/null
  146. # Add serial port and enable console output
  147. set +o errtrace
  148. (
  149. echo -e "\e[1;33m Adding serial port and configuring console... \e[0m"
  150. trap '
  151. warn "Unable to configure serial port. VM is still functional."
  152. if [ "$(qm config $VMID | sed -n ''/serial0/p'')" != "" ]; then
  153. qm set $VMID --delete serial0 >/dev/null
  154. fi
  155. exit
  156. ' ERR
  157. if [ "$(command -v kpartx)" = "" ]; then
  158. echo -e "\e[1;33m Installing kpartx... \e[0m"
  159. apt-get --allow-releaseinfo-change update >/dev/null
  160. apt-get -qqy install kpartx &>/dev/null
  161. fi
  162. DISK1_PATH="$(pvesm path $DISK1_REF)"
  163. DISK1_PART1="$(kpartx -al $DISK1_PATH | awk 'NR==1 {print $1}')"
  164. DISK1_PART1_PATH="/dev/mapper/$DISK1_PART1"
  165. TEMP_MOUNT="${TEMP_DIR}/mnt"
  166. trap '
  167. findmnt $TEMP_MOUNT >/dev/null && umount $TEMP_MOUNT
  168. command -v kpartx >/dev/null && kpartx -d $DISK1_PATH
  169. ' EXIT
  170. kpartx -a $DISK1_PATH
  171. mkdir $TEMP_MOUNT
  172. mount $DISK1_PART1_PATH $TEMP_MOUNT
  173. sed -i 's/$/ console=ttyS0/' ${TEMP_MOUNT}/cmdline.txt
  174. qm set $VMID -serial0 socket >/dev/null
  175. )
  176. info "Completed Successfully! New VM ID is \e[1m$VMID\e[0m."