# Copyright (c) 2021-2025 community-scripts ORG # License: MIT | https://raw.githubusercontent.com/community-scripts/ProxmoxVED/main/LICENSE # if ! declare -f wait_for >/dev/null; then # echo "[DEBUG] Undefined function 'wait_for' used from: ${BASH_SOURCE[*]}" >&2 # wait_for() { # echo "[DEBUG] Fallback: wait_for called with: $*" >&2 # true # } # fi # trap 'on_error $? $LINENO' ERR # trap 'on_exit' EXIT # trap 'on_interrupt' INT # trap 'on_terminate' TERM # if ! declare -f wait_for >/dev/null; then # wait_for() { # true # } # fi # declare -A MSG_INFO_SHOWN=() # SPINNER_PID="" # SPINNER_ACTIVE=0 # SPINNER_MSG="" # ------------------------------------------------------------------------------ # Loads core utility groups once (colors, formatting, icons, defaults). # ------------------------------------------------------------------------------ [[ -n "${_CORE_FUNC_LOADED:-}" ]] && return _CORE_FUNC_LOADED=1 load_functions() { [[ -n "${__FUNCTIONS_LOADED:-}" ]] && return __FUNCTIONS_LOADED=1 color formatting icons default_vars set_std_mode # add more } # ============================================================================ # Error & Signal Handling – robust, universal, subshell-safe # ============================================================================ # _stop_spinner_on_error() { # [[ -n "${SPINNER_PID:-}" ]] && kill "$SPINNER_PID" 2>/dev/null && wait "$SPINNER_PID" 2>/dev/null || true # } _tool_error_hint() { local cmd="$1" local code="$2" case "$cmd" in curl) case "$code" in 6) echo "Curl: Could not resolve host (DNS problem)" ;; 7) echo "Curl: Failed to connect to host (connection refused)" ;; 22) echo "Curl: HTTP error (404/403 etc)" ;; 28) echo "Curl: Operation timeout" ;; *) echo "Curl: Unknown error ($code)" ;; esac ;; wget) echo "Wget failed – URL unreachable or permission denied" ;; systemctl) echo "Systemd unit failure – check service name and permissions" ;; jq) echo "jq parse error – malformed JSON or missing key" ;; mariadb | mysql) echo "MySQL/MariaDB command failed – check credentials or DB" ;; unzip) echo "unzip failed – corrupt file or missing permission" ;; tar) echo "tar failed – invalid format or missing binary" ;; node | npm | pnpm | yarn) echo "Node tool failed – check version compatibility or package.json" ;; *) echo "" ;; esac } # on_error() { # local code="$?" # local line="${BASH_LINENO[0]:-unknown}" # local cmd="${BASH_COMMAND:-unknown}" # # Signalcode unterdrücken, falls INT/TERM kommt # [[ "$code" == "130" || "$code" == "143" ]] && return # _stop_spinner_on_error # msg_error "Script failed at line $line with exit code $code: $cmd" # exit "$code" # } # on_exit() { # _stop_spinner_on_error # [[ "${VERBOSE:-no}" == "yes" ]] && msg_info "Script exited cleanly" # } # on_interrupt() { # _stop_spinner_on_error # msg_error "Interrupted by user (CTRL+C)" # exit 130 # } # on_terminate() { # _stop_spinner_on_error # msg_error "Terminated by signal (SIGTERM)" # exit 143 # } catch_errors() { set -Eeuo pipefail trap 'error_handler $LINENO "$BASH_COMMAND"' ERR } # ------------------------------------------------------------------------------ # Sets ANSI color codes used for styled terminal output. # ------------------------------------------------------------------------------ color() { YW=$(echo "\033[33m") YWB=$'\e[93m' BL=$(echo "\033[36m") RD=$(echo "\033[01;31m") BGN=$(echo "\033[4;92m") GN=$(echo "\033[1;92m") DGN=$(echo "\033[32m") CL=$(echo "\033[m") } # Special for spinner and colorized output via printf color_spinner() { CS_YW=$'\033[33m' CS_YWB=$'\033[93m' CS_CL=$'\033[m' } # ------------------------------------------------------------------------------ # Defines formatting helpers like tab, bold, and line reset sequences. # ------------------------------------------------------------------------------ formatting() { BFR="\\r\\033[K" BOLD=$(echo "\033[1m") HOLD=" " TAB=" " TAB3=" " } # ------------------------------------------------------------------------------ # Sets symbolic icons used throughout user feedback and prompts. # ------------------------------------------------------------------------------ icons() { CM="${TAB}✔️${TAB}" CROSS="${TAB}✖️${TAB}" DNSOK="✔️ " DNSFAIL="${TAB}✖️${TAB}" INFO="${TAB}💡${TAB}${CL}" OS="${TAB}🖥️${TAB}${CL}" OSVERSION="${TAB}🌟${TAB}${CL}" CONTAINERTYPE="${TAB}📦${TAB}${CL}" DISKSIZE="${TAB}💾${TAB}${CL}" CPUCORE="${TAB}🧠${TAB}${CL}" RAMSIZE="${TAB}🛠️${TAB}${CL}" SEARCH="${TAB}🔍${TAB}${CL}" VERBOSE_CROPPED="🔍${TAB}" VERIFYPW="${TAB}🔐${TAB}${CL}" CONTAINERID="${TAB}🆔${TAB}${CL}" HOSTNAME="${TAB}🏠${TAB}${CL}" BRIDGE="${TAB}🌉${TAB}${CL}" NETWORK="${TAB}📡${TAB}${CL}" GATEWAY="${TAB}🌐${TAB}${CL}" DISABLEIPV6="${TAB}🚫${TAB}${CL}" DEFAULT="${TAB}⚙️${TAB}${CL}" MACADDRESS="${TAB}🔗${TAB}${CL}" VLANTAG="${TAB}🏷️${TAB}${CL}" ROOTSSH="${TAB}🔑${TAB}${CL}" CREATING="${TAB}🚀${TAB}${CL}" ADVANCED="${TAB}🧩${TAB}${CL}" FUSE="${TAB}🗂️${TAB}${CL}" HOURGLASS="${TAB}⏳${TAB}" } # ------------------------------------------------------------------------------ # Sets default retry and wait variables used for system actions. # ------------------------------------------------------------------------------ default_vars() { RETRY_NUM=10 RETRY_EVERY=3 i=$RETRY_NUM #[[ "${VAR_OS:-}" == "unknown" ]] } # ------------------------------------------------------------------------------ # Sets default verbose mode for script and os execution. # ------------------------------------------------------------------------------ set_std_mode() { if [ "${VERBOSE:-no}" = "yes" ]; then STD="" else STD="silent" fi } # Silent execution function silent() { "$@" >/dev/null 2>&1 } # Function to download & save header files get_header() { local app_name=$(echo "${APP,,}" | tr -d ' ') local app_type=${APP_TYPE:-ct} # Default zu 'ct' falls nicht gesetzt local header_url="https://git.community-scripts.org/community-scripts/ProxmoxVED/raw/branch/main/${app_type}/headers/${app_name}" local local_header_path="/usr/local/community-scripts/headers/${app_type}/${app_name}" mkdir -p "$(dirname "$local_header_path")" if [ ! -s "$local_header_path" ]; then if ! curl -fsSL "$header_url" -o "$local_header_path"; then return 1 fi fi cat "$local_header_path" 2>/dev/null || true } header_info() { local app_name=$(echo "${APP,,}" | tr -d ' ') local header_content header_content=$(get_header "$app_name") || header_content="" clear local term_width term_width=$(tput cols 2>/dev/null || echo 120) if [ -n "$header_content" ]; then echo "$header_content" fi } # ------------------------------------------------------------------------------ # Performs a curl request with retry logic and inline feedback. # ------------------------------------------------------------------------------ run_curl() { if [ "$VERBOSE" = "no" ]; then $STD curl "$@" else curl "$@" fi } curl_handler() { set +e trap 'set -e' RETURN local args=() local url="" local max_retries=3 local delay=2 local attempt=1 local exit_code local has_output_file=false local result="" # Parse arguments for arg in "$@"; do if [[ "$arg" != -* && -z "$url" ]]; then url="$arg" fi [[ "$arg" == "-o" || "$arg" == --output ]] && has_output_file=true args+=("$arg") done if [[ -z "$url" ]]; then msg_error "No valid URL or option entered for curl_handler" return 1 fi $STD msg_info "Fetching: $url" while [[ $attempt -le $max_retries ]]; do if $has_output_file; then $STD run_curl "${args[@]}" exit_code=$? else result=$(run_curl "${args[@]}") exit_code=$? fi if [[ $exit_code -eq 0 ]]; then $STD msg_ok "Fetched: $url" $has_output_file || printf '%s' "$result" return 0 fi if ((attempt >= max_retries)); then # Read error log if it exists if [ -s /tmp/curl_error.log ]; then local curl_stderr curl_stderr=$(&2 sleep "$delay" ((attempt++)) done set -e } # ------------------------------------------------------------------------------ # Handles specific curl error codes and displays descriptive messages. # ------------------------------------------------------------------------------ __curl_err_handler() { local exit_code="$1" local target="$2" local curl_msg="$3" case $exit_code in 1) msg_error "Unsupported protocol: $target" ;; 2) msg_error "Curl init failed: $target" ;; 3) msg_error "Malformed URL: $target" ;; 5) msg_error "Proxy resolution failed: $target" ;; 6) msg_error "Host resolution failed: $target" ;; 7) msg_error "Connection failed: $target" ;; 9) msg_error "Access denied: $target" ;; 18) msg_error "Partial file transfer: $target" ;; 22) msg_error "HTTP error (e.g. 400/404): $target" ;; 23) msg_error "Write error on local system: $target" ;; 26) msg_error "Read error from local file: $target" ;; 28) msg_error "Timeout: $target" ;; 35) msg_error "SSL connect error: $target" ;; 47) msg_error "Too many redirects: $target" ;; 51) msg_error "SSL cert verify failed: $target" ;; 52) msg_error "Empty server response: $target" ;; 55) msg_error "Send error: $target" ;; 56) msg_error "Receive error: $target" ;; 60) msg_error "SSL CA not trusted: $target" ;; 67) msg_error "Login denied by server: $target" ;; 78) msg_error "Remote file not found (404): $target" ;; *) msg_error "Curl failed with code $exit_code: $target" ;; esac [[ -n "$curl_msg" ]] && printf "%s\n" "$curl_msg" >&2 exit 1 } fatal() { msg_error "$1" kill -INT $$ } # # Ensure POSIX compatibility across Alpine and Debian/Ubuntu # # === Spinner Start === # # Trap cleanup on various signals # trap 'cleanup_spinner' EXIT INT TERM HUP # spinner_frames=('⠋' '⠙' '⠹' '⠸' '⠼' '⠴' '⠦' '⠧' '⠇' '⠏') # # === Spinner Start === # start_spinner() { # local msg="$1" # local spin_i=0 # local interval=0.1 # stop_spinner # SPINNER_MSG="$msg" # SPINNER_ACTIVE=1 # { # while [[ "$SPINNER_ACTIVE" -eq 1 ]]; do # if [[ -t 2 ]]; then # printf "\r\e[2K%s %b" "${TAB}${spinner_frames[spin_i]}${TAB}" "${YW}${SPINNER_MSG}${CL}" >&2 # else # printf "%s...\n" "$SPINNER_MSG" >&2 # break # fi # spin_i=$(((spin_i + 1) % ${#spinner_frames[@]})) # sleep "$interval" # done # } & # local pid=$! # if ps -p "$pid" >/dev/null 2>&1; then # SPINNER_PID="$pid" # else # SPINNER_ACTIVE=0 # SPINNER_PID="" # fi # } # # === Spinner Stop === # stop_spinner() { # if [[ "$SPINNER_ACTIVE" -eq 1 && -n "$SPINNER_PID" ]]; then # SPINNER_ACTIVE=0 # if kill -0 "$SPINNER_PID" 2>/dev/null; then # kill "$SPINNER_PID" 2>/dev/null || true # for _ in $(seq 1 10); do # sleep 0.05 # kill -0 "$SPINNER_PID" 2>/dev/null || break # done # fi # if [[ "$SPINNER_PID" =~ ^[0-9]+$ ]]; then # ps -p "$SPINNER_PID" -o pid= >/dev/null 2>&1 && wait "$SPINNER_PID" 2>/dev/null || true # fi # printf "\r\e[2K" >&2 # SPINNER_PID="" # fi # } # cleanup_spinner() { # stop_spinner # } # msg_info() { # local msg="$1" # [[ -z "$msg" || -n "${MSG_INFO_SHOWN["$msg"]+x}" ]] && return # MSG_INFO_SHOWN["$msg"]=1 # stop_spinner # echo -e "OS: ${var_os:-}" # if [[ "${VERBOSE:-no}" != "no" || "${var_os:-}" == "alpine" || ! -t 2 ]]; then # printf "\r\e[2K%s %b\n" "$HOURGLASS" "${YW}${msg}${CL}" >&2 # else # start_spinner "$msg" # fi # } spinner() { local chars=(⠋ ⠙ ⠹ ⠸ ⠼ ⠴ ⠦ ⠧ ⠇ ⠏) local i=0 while true; do local index=$((i++ % ${#chars[@]})) printf "\r\033[2K%s %b" "${CS_YWB}${chars[$index]}${CS_CL}" "${CS_YWB}${SPINNER_MSG:-}${CS_CL}" sleep 0.1 done } stop_spinner() { local pid="${SPINNER_PID:-}" [[ -z "$pid" && -f /tmp/.spinner.pid ]] && pid=$(/dev/null; then sleep 0.05 kill -9 "$pid" 2>/dev/null || true wait "$pid" 2>/dev/null || true fi rm -f /tmp/.spinner.pid unset SPINNER_PID fi stty sane 2>/dev/null #printf "\r\033[K\e[?25h\n" } msg_info() { local msg="$1" SPINNER_MSG="$msg" color_spinner spinner & SPINNER_PID=$! echo "$SPINNER_PID" >/tmp/.spinner.pid disown "$SPINNER_PID" 2>/dev/null || true } msg_ok() { stop_spinner local msg="$1" echo -e "${BFR:-} ${CM:-✔️} ${GN}${msg}${CL}" } msg_error() { stop_spinner local msg="$1" echo -e "${BFR:-} ${CROSS:-✖️} ${RD}${msg}${CL}" } msg_warn() { stop_spinner local msg="$1" echo -e "${BFR:-} ${INFO:-ℹ️} ${YWB}${msg}${CL}" } msg_custom() { local symbol="${1:-"[*]"}" local color="${2:-"\e[36m"}" local msg="${3:-}" [[ -z "$msg" ]] && return stop_spinner echo -e "${BFR:-} ${symbol} ${color}${msg}${CL:-\e[0m}" printf "\r\033[K\e[?25h\n" } # msg_ok() { # local msg="$1" # [[ -z "$msg" ]] && return # stop_spinner # printf "\r\e[2K%s %b\n" "$CM" "${GN}${msg}${CL}" >&2 # if declare -p MSG_INFO_SHOWN &>/dev/null && [[ "$(declare -p MSG_INFO_SHOWN 2>/dev/null)" =~ "declare -A" ]]; then # unset MSG_INFO_SHOWN["$msg"] # fi # } # msg_error() { # local msg="$1" # [[ -z "$msg" ]] && return # stop_spinner # printf "\r\e[2K%s %b\n" "$CROSS" "${RD}${msg}${CL}" >&2 # } # msg_warn() { # local msg="$1" # [[ -z "$msg" ]] && return # stop_spinner # printf "\r\e[2K%s %b\n" "$INFO" "${YWB}${msg}${CL}" >&2 # if declare -p MSG_INFO_SHOWN &>/dev/null && [[ "$(declare -p MSG_INFO_SHOWN 2>/dev/null)" =~ "declare -A" ]]; then # unset MSG_INFO_SHOWN["$msg"] # fi # } # msg_custom() { # local symbol="${1:-"[*]"}" # local color="${2:-"\e[36m"}" # Default: Cyan # local msg="${3:-}" # [[ -z "$msg" ]] && return # stop_spinner 2>/dev/null || true # printf "\r\e[2K%s %b\n" "$symbol" "${color}${msg}${CL:-\e[0m}" >&2 # } # msg_progress() { # local current="$1" # local total="$2" # local label="$3" # local width=40 # local filled percent bar empty # local fill_char="#" # local empty_char="-" # if ! [[ "$current" =~ ^[0-9]+$ ]] || ! [[ "$total" =~ ^[0-9]+$ ]] || [[ "$total" -eq 0 ]]; then # printf "\r\e[2K%s %b\n" "$CROSS" "${RD}Invalid progress input${CL}" >&2 # return # fi # percent=$(((current * 100) / total)) # filled=$(((current * width) / total)) # empty=$((width - filled)) # bar=$(printf "%${filled}s" | tr ' ' "$fill_char") # bar+=$(printf "%${empty}s" | tr ' ' "$empty_char") # printf "\r\e[2K%s [%s] %3d%% %s" "${TAB}" "$bar" "$percent" "$label" >&2 # if [[ "$current" -eq "$total" ]]; then # printf "\n" >&2 # fi # } run_container_safe() { local ct="$1" shift local cmd="$*" lxc-attach -n "$ct" -- bash -euo pipefail -c " trap 'echo Aborted in container; exit 130' SIGINT SIGTERM $cmd " || __handle_general_error "lxc-attach to CT $ct" } check_or_create_swap() { msg_info "Checking for active swap" if swapon --noheadings --show | grep -q 'swap'; then msg_ok "Swap is active" return 0 fi msg_error "No active swap detected" read -p "Do you want to create a swap file? [y/N]: " create_swap create_swap="${create_swap,,}" # to lowercase if [[ "$create_swap" != "y" && "$create_swap" != "yes" ]]; then msg_info "Skipping swap file creation" return 1 fi read -p "Enter swap size in MB (e.g., 2048 for 2GB): " swap_size_mb if ! [[ "$swap_size_mb" =~ ^[0-9]+$ ]]; then msg_error "Invalid size input. Aborting." return 1 fi local swap_file="/swapfile" msg_info "Creating ${swap_size_mb}MB swap file at $swap_file" if dd if=/dev/zero of="$swap_file" bs=1M count="$swap_size_mb" status=progress && chmod 600 "$swap_file" && mkswap "$swap_file" && swapon "$swap_file"; then msg_ok "Swap file created and activated successfully" else msg_error "Failed to create or activate swap" return 1 fi } trap 'stop_spinner' EXIT INT TERM