adguard_container.sh 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. #!/usr/bin/env bash
  2. while true; do
  3. read -p "This will create a New AdGuard Home LXC. Proceed(y/n)?" yn
  4. case $yn in
  5. [Yy]* ) break;;
  6. [Nn]* ) exit;;
  7. * ) echo "Please answer yes or no.";;
  8. esac
  9. done
  10. set -o errexit
  11. set -o errtrace
  12. set -o nounset
  13. set -o pipefail
  14. shopt -s expand_aliases
  15. alias die='EXIT=$? LINE=$LINENO error_exit'
  16. trap die ERR
  17. trap cleanup EXIT
  18. function error_exit() {
  19. trap - ERR
  20. local DEFAULT='Unknown failure occured.'
  21. local REASON="\e[97m${1:-$DEFAULT}\e[39m"
  22. local FLAG="\e[91m[ERROR] \e[93m$EXIT@$LINE"
  23. msg "$FLAG $REASON"
  24. [ ! -z ${CTID-} ] && cleanup_ctid
  25. exit $EXIT
  26. }
  27. function warn() {
  28. local REASON="\e[97m$1\e[39m"
  29. local FLAG="\e[93m[WARNING]\e[39m"
  30. msg "$FLAG $REASON"
  31. }
  32. function info() {
  33. local REASON="$1"
  34. local FLAG="\e[36m[INFO]\e[39m"
  35. msg "$FLAG $REASON"
  36. }
  37. function msg() {
  38. local TEXT="$1"
  39. echo -e "$TEXT"
  40. }
  41. function cleanup_ctid() {
  42. if [ ! -z ${MOUNT+x} ]; then
  43. pct unmount $CTID
  44. fi
  45. if $(pct status $CTID &>/dev/null); then
  46. if [ "$(pct status $CTID | awk '{print $2}')" == "running" ]; then
  47. pct stop $CTID
  48. fi
  49. pct destroy $CTID
  50. elif [ "$(pvesm list $STORAGE --vmid $CTID)" != "" ]; then
  51. pvesm free $ROOTFS
  52. fi
  53. }
  54. function cleanup() {
  55. popd >/dev/null
  56. rm -rf $TEMP_DIR
  57. }
  58. function load_module() {
  59. if ! $(lsmod | grep -Fq $1); then
  60. modprobe $1 &>/dev/null || \
  61. die "Failed to load '$1' module."
  62. fi
  63. MODULES_PATH=/etc/modules
  64. if ! $(grep -Fxq "$1" $MODULES_PATH); then
  65. echo "$1" >> $MODULES_PATH || \
  66. die "Failed to add '$1' module to load at boot."
  67. fi
  68. }
  69. TEMP_DIR=$(mktemp -d)
  70. pushd $TEMP_DIR >/dev/null
  71. wget -qL https://raw.githubusercontent.com/tteck/Proxmox/main/setup/adguard_setup.sh
  72. load_module overlay
  73. while read -r line; do
  74. TAG=$(echo $line | awk '{print $1}')
  75. TYPE=$(echo $line | awk '{printf "%-10s", $2}')
  76. FREE=$(echo $line | numfmt --field 4-6 --from-unit=K --to=iec --format %.2f | awk '{printf( "%9sB", $6)}')
  77. ITEM=" Type: $TYPE Free: $FREE "
  78. OFFSET=2
  79. if [[ $((${#ITEM} + $OFFSET)) -gt ${MSG_MAX_LENGTH:-} ]]; then
  80. MSG_MAX_LENGTH=$((${#ITEM} + $OFFSET))
  81. fi
  82. STORAGE_MENU+=( "$TAG" "$ITEM" "OFF" )
  83. done < <(pvesm status -content rootdir | awk 'NR>1')
  84. if [ $((${#STORAGE_MENU[@]}/3)) -eq 0 ]; then
  85. warn "'Container' needs to be selected for at least one storage location."
  86. die "Unable to detect valid storage location."
  87. elif [ $((${#STORAGE_MENU[@]}/3)) -eq 1 ]; then
  88. STORAGE=${STORAGE_MENU[0]}
  89. else
  90. while [ -z "${STORAGE:+x}" ]; do
  91. STORAGE=$(whiptail --title "Storage Pools" --radiolist \
  92. "Which storage pool you would like to use for the container?\n\n" \
  93. 16 $(($MSG_MAX_LENGTH + 23)) 6 \
  94. "${STORAGE_MENU[@]}" 3>&1 1>&2 2>&3) || exit
  95. done
  96. fi
  97. info "Using '$STORAGE' for storage location."
  98. CTID=$(pvesh get /cluster/nextid)
  99. info "LXC ID is $CTID."
  100. msg "Updating LXC template list..."
  101. pveam update >/dev/null
  102. msg "Downloading LXC template..."
  103. OSTYPE=debian
  104. OSVERSION=${OSTYPE}-11
  105. mapfile -t TEMPLATES < <(pveam available -section system | sed -n "s/.*\($OSVERSION.*\)/\1/p" | sort -t - -k 2 -V)
  106. TEMPLATE="${TEMPLATES[-1]}"
  107. pveam download local $TEMPLATE >/dev/null ||
  108. die "A problem occured while downloading the LXC template."
  109. STORAGE_TYPE=$(pvesm status -storage $STORAGE | awk 'NR>1 {print $2}')
  110. case $STORAGE_TYPE in
  111. dir|nfs)
  112. DISK_EXT=".raw"
  113. DISK_REF="$CTID/"
  114. ;;
  115. zfspool)
  116. DISK_PREFIX="subvol"
  117. DISK_FORMAT="subvol"
  118. ;;
  119. esac
  120. DISK=${DISK_PREFIX:-vm}-${CTID}-disk-0${DISK_EXT-}
  121. ROOTFS=${STORAGE}:${DISK_REF-}${DISK}
  122. msg "Creating LXC..."
  123. DISK_SIZE=2G
  124. pvesm alloc $STORAGE $CTID $DISK $DISK_SIZE --format ${DISK_FORMAT:-raw} >/dev/null
  125. if [ "$STORAGE_TYPE" == "zfspool" ]; then
  126. warn "Some containers may not work properly due to ZFS not supporting 'fallocate'."
  127. else
  128. mkfs.ext4 $(pvesm path $ROOTFS) &>/dev/null
  129. fi
  130. ARCH=$(dpkg --print-architecture)
  131. HOSTNAME=adguard
  132. TEMPLATE_STRING="local:vztmpl/${TEMPLATE}"
  133. pct create $CTID $TEMPLATE_STRING -arch $ARCH -features nesting=1 \
  134. -hostname $HOSTNAME -net0 name=eth0,bridge=vmbr0,ip=dhcp -onboot 1 -cores 1 -memory 512 \
  135. -ostype $OSTYPE -rootfs $ROOTFS,size=$DISK_SIZE -storage $STORAGE >/dev/null
  136. MOUNT=$(pct mount $CTID | cut -d"'" -f 2)
  137. ln -fs $(readlink /etc/localtime) ${MOUNT}/etc/localtime
  138. pct unmount $CTID && unset MOUNT
  139. msg "Starting LXC..."
  140. pct start $CTID
  141. pct push $CTID adguard_setup.sh /adguard_setup.sh -perms 755
  142. pct exec $CTID /adguard_setup.sh
  143. IP=$(pct exec $CTID ip a s dev eth0 | sed -n '/inet / s/\// /p' | awk '{print $2}')
  144. info "Successfully created a AdGuard Home LXC to $CTID at IP Address ${IP}"