#!/usr/bin/env bash
NEXTID = $( pvesh get /cluster/nextid)
INTEGER = '^[0-9]+$'
YW = ` echo "\033[33m" `
BL = ` echo "\033[36m" `
RD = ` echo "\033[01;31m" `
CM = '\xE2\x9C\x94\033'
BGN = ` echo "\033[4;92m" `
GN = ` echo "\033[32m" `
CL = ` echo "\033[m" `
APP = "Zigbee2MQTT"
NSAPP = $( echo ${ APP ,, } | tr -d ' ' )
while true; do
read -p " This will create a New ${ APP } LXC. Proceed(y/n)? " yn
case $yn in
[ Yy] * ) break; ;
[ Nn] * ) exit; ;
* ) echo "Please answer yes or no." ; ;
esac
done
clear
function header_info {
echo -e " ${ YW }
_______ _ ___ __ __ ____ _______ _______
| ___ ( _) | | | __ \| \/ | / __ \_ _ __| __ __|
/ / _ __ _| | __ ___ ___ ) | \ / | | | | | | | |
/ / | | / _ | _ \ / _ \/ _ \ / /| | \/ | | | | | | | | |
/ /__| | ( _| | | _) | __/ __// /_| | | | | __| | | | | |
/_____| _| \_ _, | ____/ \_ __| \_ __| ____| _| | _| \_ __\_ \ | _| | _|
v3 __/ |
| ___/
${ CL } "
}
header_info
function pve_check( ) {
PVE = $( pveversion | grep "pve-manager/7" | wc -l)
if [ [ $PVE != 1 ] ] ; then
echo -e " ${ RD } This script requires Proxmox Virtual Environment 7.0 or greater "
echo -e "Exiting..."
sleep 2
exit
fi
}
function default_settings( ) {
clear
header_info
echo -e " ${ BL } Using Default Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } Privileged ${ CL } "
CT_TYPE = "0"
echo -e " ${ GN } Using CT Password ${ BGN } Automatic Login ${ CL } "
PW = " "
echo -e " ${ GN } Using ID ${ BGN } $NEXTID ${ CL } "
CT_ID = $NEXTID
echo -e " ${ GN } Using CT Name ${ BGN } $NSAPP ${ CL } "
HN = $NSAPP
echo -e " ${ GN } Using Disk Size ${ BGN } 4GB ${ CL } "
SIZEDISK = "4"
echo -e " ${ GN } Using Storage ${ BGN } local-lvm ${ CL } "
STORAGETYPE = "local-lvm"
echo -e " ${ GN } Using ${ BGN } 2vCPU ${ CL } "
CORE_COUNT = "2"
echo -e " ${ GN } Using ${ BGN } 1024MiB ${ CL } ${ GN } RAM ${ CL } "
RAM_SIZE = "1024"
echo -e " ${ GN } Using IP Address ${ BGN } DHCP ${ CL } "
NET = dhcp
echo -e " ${ GN } Using VLAN Tag ${ BGN } NONE ${ CL } "
VLAN = " "
}
function advanced_settings( ) {
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ YW } Type Unprivileged, or Press [ENTER] for Default: Privileged "
read CT_TYPE1
if [ -z $CT_TYPE1 ] ; then CT_TYPE1 = "Privileged"
CT_TYPE = "0"
echo -en " ${ GN } Set CT Type ${ BL } $CT_TYPE1 ${ CL } "
else
CT_TYPE1 = "Unprivileged"
CT_TYPE = "1"
echo -en " ${ GN } Set CT Type ${ BL } Unprivileged ${ CL } ${ RD } NO DEVICE PASSTHROUGH ${ CL } "
fi ;
echo -e " ${ CM } ${ CL } \r "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ YW } Set Password, or Press [ENTER] for Default: Automatic Login "
read PW1
if [ -z $PW1 ] ; then PW1 = "Automatic Login" PW = " " ;
echo -en " ${ GN } Set CT ${ BL } $PW1 ${ CL } "
else
PW = " -password $PW1 "
echo -en " ${ GN } Set CT Password ${ BL } $PW1 ${ CL } "
fi ;
echo -e " ${ CM } ${ CL } \r "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ YW } Enter the CT ID, or Press [ENTER] to automatically generate ( ${ NEXTID } ) "
read CT_ID
if [ -z $CT_ID ] ; then CT_ID = $NEXTID ; fi ;
echo -en " ${ GN } Set CT ID To ${ BL } $CT_ID ${ CL } "
echo -e " ${ CM } ${ CL } \r "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ YW } Enter CT Name (no-spaces), or Press [ENTER] for Default: $NSAPP "
read CT_NAME
if [ -z $CT_NAME ] ; then
HN = $NSAPP
else
HN = $( echo ${ CT_NAME ,, } | tr -d ' ' )
fi
echo -en " ${ GN } Set CT Name To ${ BL } $HN ${ CL } "
echo -e " ${ CM } ${ CL } \r "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ GN } Using CT Name ${ BGN } $HN ${ CL } "
echo -e " ${ YW } Enter a Disk Size, or Press [ENTER] for Default: 4Gb "
read SIZEDISK
if [ -z $SIZEDISK ] ; then SIZEDISK = "4" ; fi ;
if ! [ [ $SIZEDISK = ~ $INTEGER ] ] ; then echo "ERROR! SIZEDISK MUST HAVE INTEGER NUMBER!" ; exit; fi ;
echo -en " ${ GN } Set Disk Size To ${ BL } $SIZEDISK ${ CL } "
echo -e " ${ CM } ${ CL } \r "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ GN } Using CT Name ${ BGN } $HN ${ CL } "
echo -e " ${ GN } Using Disk Size ${ BGN } $SIZEDISK ${ CL } "
echo -e " ${ YW } Storages Available: ${ CL } "
echo " "
for stg in ` pvesh get storage --noborder --noheader`
do
echo -e " ${ BL } - ${ stg } ${ CL } "
done
echo " "
echo -e " ${ YW } Enter which storage to create the CT, or Press [ENTER] for Default: local-lvm "
read STORAGETYPE
if [ -z $STORAGETYPE ] ; then STORAGETYPE = "local-lvm" ; fi ;
echo -en " ${ GN } Set Storage To ${ BL } $STORAGETYPE ${ CL } "
echo -e " ${ CM } ${ CL } \r "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ GN } Using CT Name ${ BGN } $HN ${ CL } "
echo -e " ${ GN } Using Disk Size ${ BGN } $SIZEDISK ${ CL } "
echo -e " ${ GN } Using Storage ${ BGN } $STORAGETYPE ${ CL } "
echo -e " ${ YW } Allocate CPU cores, or Press [ENTER] for Default: 2 "
read CORE_COUNT
if [ -z $CORE_COUNT ] ; then CORE_COUNT = "2" ; fi ;
echo -en " ${ GN } Set Cores To ${ BL } $CORE_COUNT ${ CL } "
echo -e " ${ CM } ${ CL } \r "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ GN } Using CT Name ${ BGN } $HN ${ CL } "
echo -e " ${ GN } Using Disk Size ${ BGN } $SIZEDISK ${ CL } "
echo -e " ${ GN } Using Storage ${ BGN } $STORAGETYPE ${ CL } "
echo -e " ${ GN } Using ${ BGN } ${ CORE_COUNT } vCPU ${ CL } "
echo -e " ${ YW } Allocate RAM in MiB, or Press [ENTER] for Default: 1024 "
read RAM_SIZE
if [ -z $RAM_SIZE ] ; then RAM_SIZE = "1024" ; fi ;
echo -en " ${ GN } Set RAM To ${ BL } $RAM_SIZE ${ CL } "
echo -e " ${ CM } ${ CL } \n "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ GN } Using CT Name ${ BGN } $HN ${ CL } "
echo -e " ${ GN } Using Disk Size ${ BGN } $SIZEDISK ${ CL } "
echo -e " ${ GN } Using Storage ${ BGN } $STORAGETYPE ${ CL } "
echo -e " ${ GN } Using ${ BGN } ${ CORE_COUNT } vCPU ${ CL } "
echo -e " ${ GN } Using ${ BGN } ${ RAM_SIZE } MiB ${ CL } ${ GN } RAM ${ CL } "
echo -e " ${ YW } Enter a IP Address, or Press [ENTER] for Default: DHCP "
read NET
if [ -z $NET ] ; then NET = "dhcp" ; fi ;
echo -en " ${ GN } Set IP Address To ${ BL } $NET ${ CL } "
echo -e " ${ CM } ${ CL } \n "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ GN } Using CT Name ${ BGN } $HN ${ CL } "
echo -e " ${ GN } Using Disk Size ${ BGN } $SIZEDISK ${ CL } "
echo -e " ${ GN } Using Storage ${ BGN } $STORAGETYPE ${ CL } "
echo -e " ${ GN } Using ${ BGN } ${ CORE_COUNT } vCPU ${ CL } "
echo -e " ${ GN } Using ${ BGN } ${ RAM_SIZE } MiB ${ CL } ${ GN } RAM ${ CL } "
echo -e " ${ GN } Using IP Address ${ BGN } $NET ${ CL } "
echo -e " ${ YW } Enter a VLAN Tag, or Press [ENTER] for Default: NONE "
read VLAN1
if [ -z $VLAN1 ] ; then VLAN1 = "NONE" VLAN = " " ;
echo -en " ${ GN } Set VLAN Tag To ${ BL } $VLAN1 ${ CL } "
else
VLAN = " -tag $VLAN1 "
echo -en " ${ GN } Set VLAN Tag To ${ BL } $VLAN1 ${ CL } "
fi ;
echo -e " ${ CM } ${ CL } \n "
sleep 1
clear
header_info
echo -e " ${ RD } Using Advanced Settings ${ CL } "
echo -e " ${ GN } Using CT Type ${ BGN } $CT_TYPE1 ${ CL } "
echo -e " ${ GN } Using CT Password ${ BGN } $PW1 ${ CL } "
echo -e " ${ GN } Using ID ${ BGN } $CT_ID ${ CL } "
echo -e " ${ GN } Using CT Name ${ BGN } $HN ${ CL } "
echo -e " ${ GN } Using Disk Size ${ BGN } $SIZEDISK ${ CL } "
echo -e " ${ GN } Using Storage ${ BGN } $STORAGETYPE ${ CL } "
echo -e " ${ GN } Using ${ BGN } ${ CORE_COUNT } vCPU ${ CL } "
echo -e " ${ GN } Using ${ BGN } ${ RAM_SIZE } MiB ${ CL } ${ GN } RAM ${ CL } "
echo -e " ${ GN } Using IP Address ${ BGN } $NET ${ CL } "
echo -e " ${ GN } Using VLAN Tag ${ BGN } $VLAN1 ${ CL } "
read -p "Are these settings correct(y/n)? " -n 1 -r
echo
if [ [ ! $REPLY = ~ ^[ Yy] $ ] ]
then
advanced_settings
fi
}
function start_script( ) {
echo -e " ${ YW } Type Advanced, or Press [ENTER] for Default Settings "
read SETTINGS
if [ -z $SETTINGS ] ; then default_settings;
else
advanced_settings
fi ;
}
pve_check
start_script
set -o errexit
set -o errtrace
set -o nounset
set -o pipefail
shopt -s expand_aliases
alias die = 'EXIT=$? LINE=$LINENO error_exit'
trap die ERR
trap cleanup EXIT
function error_exit( ) {
trap - ERR
local DEFAULT = 'Unknown failure occured.'
local REASON = " \e[97m ${ 1 :- $DEFAULT } \e[39m "
local FLAG = " \e[91m[ERROR] \e[93m $EXIT @ $LINE "
msg " $FLAG $REASON "
[ ! -z ${ CTID - } ] && cleanup_ctid
exit $EXIT
}
function warn( ) {
local REASON = " \e[97m $1 \e[39m "
local FLAG = "\e[93m[WARNING]\e[39m"
msg " $FLAG $REASON "
}
function info( ) {
local REASON = " $1 "
local FLAG = "\e[36m[INFO]\e[39m"
msg " $FLAG $REASON "
}
function msg( ) {
local TEXT = " $1 "
echo -e " $TEXT "
}
function cleanup_ctid( ) {
if $( pct status $CTID & >/dev/null) ; then
if [ " $( pct status $CTID | awk '{print $2}' ) " = = "running" ] ; then
pct stop $CTID
fi
pct destroy $CTID
elif [ " $( pvesm list $STORAGE --vmid $CTID ) " != "" ] ; then
pvesm free $ROOTFS
fi
}
function cleanup( ) {
popd >/dev/null
rm -rf $TEMP_DIR
}
if [ " $CT_TYPE " = = "1" ] ; then
FEATURES = "nesting=1,keyctl=1"
else
FEATURES = "nesting=1"
fi
TEMP_DIR = $( mktemp -d)
pushd $TEMP_DIR >/dev/null
export CTID = $CT_ID
export PCT_OSTYPE = debian
export PCT_OSVERSION = 11
export PCT_DISK_SIZE = $SIZEDISK
export PCT_OPTIONS = "
-features $FEATURES
-hostname $HN
-net0 name = eth0,bridge= vmbr0,ip= $NET
$VLAN
-onboot 1
-cores $CORE_COUNT
-memory $RAM_SIZE
-unprivileged $CT_TYPE
$PW
"
bash -c " $( wget -qLO - https://raw.githubusercontent.com/tteck/Proxmox/main/ct/create_lxc.sh) " || exit
STORAGE_TYPE = $( pvesm status -storage $( pct config $CTID | grep rootfs | awk -F ":" '{print $2}' ) | awk 'NR>1 {print $2}' )
if [ " $STORAGE_TYPE " = = "zfspool" ] ; then
warn "Some addons may not work due to ZFS not supporting 'fallocate'."
fi
LXC_CONFIG = /etc/pve/lxc/${ CTID } .conf
cat <<EOF >> $LXC_CONFIG
lxc.cgroup2.devices.allow: a
lxc.cap.drop:
lxc.cgroup2.devices.allow: c 188:* rwm
lxc.cgroup2.devices.allow: c 189:* rwm
lxc.mount.entry: /dev/serial/by-id dev/serial/by-id none bind,optional,create= dir
lxc.mount.entry: /dev/ttyUSB0 dev/ttyUSB0 none bind,optional,create= file
lxc.mount.entry: /dev/ttyACM0 dev/ttyACM0 none bind,optional,create= file
EOF
echo -en " ${ GN } Starting LXC Container... "
pct start $CTID
echo -e " ${ CM } ${ CL } \r "
alias lxc-cmd= " lxc-attach -n $CTID -- "
lxc-cmd bash -c " $( wget -qLO - https://raw.githubusercontent.com/tteck/Proxmox/main/setup/zigbee2mqtt-install.sh) " || exit
IP = $( pct exec $CTID ip a s dev eth0 | sed -n '/inet / s/\// /p' | awk '{print $2}' )
echo -e " ${ GN } Successfully created ${ APP } LXC to ${ CL } ${ BL } $CTID ${ CL } . \n "