mirror of
https://github.com/community-scripts/ProxmoxVE.git
synced 2026-05-23 17:04:58 +02:00
Compare commits
24 Commits
arm64-gith
...
arm64-buil
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
88a6b07334 | ||
|
|
2c1f216189 | ||
|
|
16ac4407e2 | ||
|
|
39c514a1e3 | ||
|
|
5f6a02986d | ||
|
|
079afa4600 | ||
|
|
0aeaa5ae97 | ||
|
|
9f7b951ec6 | ||
|
|
314b1b843b | ||
|
|
73e121b679 | ||
|
|
24f1347990 | ||
|
|
866b6950c0 | ||
|
|
44ec223d20 | ||
|
|
4ab3a24d03 | ||
|
|
231945dfa7 | ||
|
|
7c051fb648 | ||
|
|
6f8aa6eadc | ||
|
|
35b3b93ca6 | ||
|
|
78979189c1 | ||
|
|
a6a83b9541 | ||
|
|
e4db6be257 | ||
|
|
b9d401b178 | ||
|
|
6f62656c96 | ||
|
|
087f817bf6 |
@@ -196,7 +196,7 @@ explain_exit_code() {
|
|||||||
103) echo "Validation: Shell is not Bash" ;;
|
103) echo "Validation: Shell is not Bash" ;;
|
||||||
104) echo "Validation: Not running as root (or invoked via sudo)" ;;
|
104) echo "Validation: Not running as root (or invoked via sudo)" ;;
|
||||||
105) echo "Validation: Proxmox VE version not supported" ;;
|
105) echo "Validation: Proxmox VE version not supported" ;;
|
||||||
106) echo "Validation: Architecture not supported (ARM / PiMox)" ;;
|
106) echo "Validation: Unsupported architecture (requires amd64 or arm64)" ;;
|
||||||
107) echo "Validation: Kernel key parameters unreadable" ;;
|
107) echo "Validation: Kernel key parameters unreadable" ;;
|
||||||
108) echo "Validation: Kernel key limits exceeded" ;;
|
108) echo "Validation: Kernel key limits exceeded" ;;
|
||||||
109) echo "Proxmox: No available container ID after max attempts" ;;
|
109) echo "Proxmox: No available container ID after max attempts" ;;
|
||||||
|
|||||||
173
misc/build.func
173
misc/build.func
@@ -52,6 +52,11 @@ variables() {
|
|||||||
# as "/tmp/${NSAPP}-${CTID}-${SESSION_ID}.log" (requires CTID, not available here)
|
# as "/tmp/${NSAPP}-${CTID}-${SESSION_ID}.log" (requires CTID, not available here)
|
||||||
CTTYPE="${CTTYPE:-${CT_TYPE:-1}}"
|
CTTYPE="${CTTYPE:-${CT_TYPE:-1}}"
|
||||||
|
|
||||||
|
# ARM64 Template default variables
|
||||||
|
DEBIAN_DEFAULT_CODENAME="trixie"
|
||||||
|
UBUNTU_DEFAULT_CODENAME="noble"
|
||||||
|
ALPINE_DEFAULT_VERSION="3.23"
|
||||||
|
|
||||||
# Parse dev_mode early
|
# Parse dev_mode early
|
||||||
parse_dev_mode
|
parse_dev_mode
|
||||||
|
|
||||||
@@ -1942,7 +1947,7 @@ advanced_settings() {
|
|||||||
|
|
||||||
# ═══════════════════════════════════════════════════════════════════════════
|
# ═══════════════════════════════════════════════════════════════════════════
|
||||||
# STEP 2: Root Password
|
# STEP 2: Root Password
|
||||||
# ════════════════════════════════════════<EFBFBD><EFBFBD><EFBFBD>═══════════════════════════════<EFBFBD><EFBFBD><EFBFBD>══
|
# ═══════════════════════════════════════════════════════════════════════════
|
||||||
2)
|
2)
|
||||||
if PW1=$(whiptail --backtitle "Proxmox VE Helper Scripts [Step $STEP/$MAX_STEP]" \
|
if PW1=$(whiptail --backtitle "Proxmox VE Helper Scripts [Step $STEP/$MAX_STEP]" \
|
||||||
--title "ROOT PASSWORD" \
|
--title "ROOT PASSWORD" \
|
||||||
@@ -3052,6 +3057,9 @@ echo_default() {
|
|||||||
echo -e "${DISKSIZE}${BOLD}${DGN}Disk Size: ${BGN}${DISK_SIZE} GB${CL}"
|
echo -e "${DISKSIZE}${BOLD}${DGN}Disk Size: ${BGN}${DISK_SIZE} GB${CL}"
|
||||||
echo -e "${CPUCORE}${BOLD}${DGN}CPU Cores: ${BGN}${CORE_COUNT}${CL}"
|
echo -e "${CPUCORE}${BOLD}${DGN}CPU Cores: ${BGN}${CORE_COUNT}${CL}"
|
||||||
echo -e "${RAMSIZE}${BOLD}${DGN}RAM Size: ${BGN}${RAM_SIZE} MiB${CL}"
|
echo -e "${RAMSIZE}${BOLD}${DGN}RAM Size: ${BGN}${RAM_SIZE} MiB${CL}"
|
||||||
|
if [[ "$(dpkg --print-architecture)" == "arm64" ]]; then
|
||||||
|
echo -e "${INFO}${BOLD}${DGN}Architecture: ${BGN}arm64${CL}"
|
||||||
|
fi
|
||||||
if [[ -n "${var_gpu:-}" && "${var_gpu}" == "yes" ]]; then
|
if [[ -n "${var_gpu:-}" && "${var_gpu}" == "yes" ]]; then
|
||||||
echo -e "${GPU}${BOLD}${DGN}GPU Passthrough: ${BGN}Enabled${CL}"
|
echo -e "${GPU}${BOLD}${DGN}GPU Passthrough: ${BGN}Enabled${CL}"
|
||||||
fi
|
fi
|
||||||
@@ -3088,7 +3096,9 @@ install_script() {
|
|||||||
pve_check
|
pve_check
|
||||||
shell_check
|
shell_check
|
||||||
root_check
|
root_check
|
||||||
|
ensure_whiptail
|
||||||
arch_check
|
arch_check
|
||||||
|
arm64_notice
|
||||||
ssh_check
|
ssh_check
|
||||||
maxkeys_check
|
maxkeys_check
|
||||||
diagnostics_check
|
diagnostics_check
|
||||||
@@ -4351,19 +4361,23 @@ EOF
|
|||||||
msg_warn "Skipping timezone setup – zone '$tz' not found in container"
|
msg_warn "Skipping timezone setup – zone '$tz' not found in container"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
local _base_pkgs="sudo curl mc gnupg2 jq"
|
||||||
|
if [[ "${ARCH:-amd64}" == "arm64" ]]; then
|
||||||
|
_base_pkgs+=" openssh-server wget gcc"
|
||||||
|
fi
|
||||||
|
|
||||||
# Detect broken DNS resolver (e.g. Tailscale MagicDNS) and inject public DNS
|
# Detect broken DNS resolver (e.g. Tailscale MagicDNS) and inject public DNS
|
||||||
if ! pct exec "$CTID" -- bash -c "getent hosts deb.debian.org >/dev/null 2>&1 && getent hosts archive.ubuntu.com >/dev/null 2>&1"; then
|
if ! pct exec "$CTID" -- bash -c "getent hosts deb.debian.org >/dev/null 2>&1 && getent hosts archive.ubuntu.com >/dev/null 2>&1"; then
|
||||||
msg_warn "APT repository DNS resolution failed in container, injecting public DNS servers"
|
msg_warn "APT repository DNS resolution failed in container, injecting public DNS servers"
|
||||||
pct exec "$CTID" -- bash -c "echo -e 'nameserver 8.8.8.8\nnameserver 1.1.1.1' >/etc/resolv.conf"
|
pct exec "$CTID" -- bash -c "echo -e 'nameserver 8.8.8.8\nnameserver 1.1.1.1' >/etc/resolv.conf"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
pct exec "$CTID" -- bash -c "apt-get update 2>&1 && apt-get install -y sudo curl mc gnupg2 jq 2>&1" >>"$BUILD_LOG" 2>&1 || {
|
pct exec "$CTID" -- bash -c "apt-get update 2>&1 && apt-get install -y ${_base_pkgs} 2>&1" >>"$BUILD_LOG" 2>&1 || {
|
||||||
local failed_mirror
|
local failed_mirror
|
||||||
failed_mirror=$(pct exec "$CTID" -- bash -c "grep -m1 -oP '(?<=URIs: https?://)[^/]+' /etc/apt/sources.list.d/debian.sources 2>/dev/null || grep -m1 -oP '(?<=deb https?://)[^/]+' /etc/apt/sources.list 2>/dev/null" 2>/dev/null || echo "unknown")
|
failed_mirror=$(pct exec "$CTID" -- bash -c "grep -m1 -oP '(?<=URIs: https?://)[^/]+' /etc/apt/sources.list.d/debian.sources 2>/dev/null || grep -m1 -oP '(?<=deb https?://)[^/]+' /etc/apt/sources.list 2>/dev/null" 2>/dev/null || echo "unknown")
|
||||||
msg_warn "apt-get update failed (${failed_mirror}), trying alternate mirrors..."
|
msg_warn "apt-get update failed (${failed_mirror}), trying alternate mirrors..."
|
||||||
local mirror_exit=0
|
local mirror_exit=0
|
||||||
pct exec "$CTID" -- bash -c '
|
pct exec "$CTID" -- env APT_BASE="$_base_pkgs" bash -c '
|
||||||
APT_BASE="sudo curl mc gnupg2 jq"
|
|
||||||
DISTRO=$(. /etc/os-release 2>/dev/null && echo "$ID" || echo "debian")
|
DISTRO=$(. /etc/os-release 2>/dev/null && echo "$ID" || echo "debian")
|
||||||
|
|
||||||
if [ "$DISTRO" = "ubuntu" ]; then
|
if [ "$DISTRO" = "ubuntu" ]; then
|
||||||
@@ -4473,7 +4487,7 @@ EOF
|
|||||||
[ -f \"\$src\" ] && sed -i \"s|URIs: http[s]*://[^/]*/|URIs: http://${custom_mirror}/|g; s|deb http[s]*://[^/]*/|deb http://${custom_mirror}/|g\" \"\$src\"
|
[ -f \"\$src\" ] && sed -i \"s|URIs: http[s]*://[^/]*/|URIs: http://${custom_mirror}/|g; s|deb http[s]*://[^/]*/|deb http://${custom_mirror}/|g\" \"\$src\"
|
||||||
done
|
done
|
||||||
rm -rf /var/lib/apt/lists/*
|
rm -rf /var/lib/apt/lists/*
|
||||||
apt-get update >/dev/null 2>&1 && apt-get install -y sudo curl mc gnupg2 jq >/dev/null 2>&1
|
apt-get update >/dev/null 2>&1 && apt-get install -y ${_base_pkgs} >/dev/null 2>&1
|
||||||
" && break
|
" && break
|
||||||
msg_warn "Mirror '${custom_mirror}' also failed. Try another or type 'skip'."
|
msg_warn "Mirror '${custom_mirror}' also failed. Try another or type 'skip'."
|
||||||
done
|
done
|
||||||
@@ -4513,7 +4527,9 @@ EOF
|
|||||||
# that sends "configuring" status AFTER the host already reported "failed"
|
# that sends "configuring" status AFTER the host already reported "failed"
|
||||||
export CONTAINER_INSTALLING=true
|
export CONTAINER_INSTALLING=true
|
||||||
|
|
||||||
lxc-attach -n "$CTID" -- bash -c "$(curl -fsSL https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/install/${var_install}.sh)"
|
local _install_script
|
||||||
|
_install_script="$(curl -fsSL "https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/install/${var_install}.sh")"
|
||||||
|
lxc-attach -n "$CTID" -- bash -c "$_install_script"
|
||||||
local lxc_exit=$?
|
local lxc_exit=$?
|
||||||
|
|
||||||
unset CONTAINER_INSTALLING
|
unset CONTAINER_INSTALLING
|
||||||
@@ -4908,7 +4924,9 @@ EOF
|
|||||||
# Re-run install script in existing container (don't destroy/recreate)
|
# Re-run install script in existing container (don't destroy/recreate)
|
||||||
set +Eeuo pipefail
|
set +Eeuo pipefail
|
||||||
trap - ERR
|
trap - ERR
|
||||||
lxc-attach -n "$CTID" -- bash -c "$(curl -fsSL https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/install/${var_install}.sh)"
|
local _install_script
|
||||||
|
_install_script="$(curl -fsSL "https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/install/${var_install}.sh")"
|
||||||
|
lxc-attach -n "$CTID" -- bash -c "$_install_script"
|
||||||
local apt_retry_exit=$?
|
local apt_retry_exit=$?
|
||||||
set -Eeuo pipefail
|
set -Eeuo pipefail
|
||||||
trap 'error_handler' ERR
|
trap 'error_handler' ERR
|
||||||
@@ -5666,6 +5684,72 @@ create_lxc_container() {
|
|||||||
esac
|
esac
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ARCH="$(dpkg --print-architecture)"
|
||||||
|
|
||||||
|
# Maps OS type + version to the release variant name used by ARM64 template sources.
|
||||||
|
arm64_template_variant() {
|
||||||
|
case "$1:$2" in
|
||||||
|
debian:12) echo "bookworm" ;;
|
||||||
|
debian:13) echo "trixie" ;;
|
||||||
|
debian:) echo "$DEBIAN_DEFAULT_CODENAME" ;;
|
||||||
|
|
||||||
|
ubuntu:24.04) echo "noble" ;;
|
||||||
|
ubuntu:26.04) echo "questing" ;;
|
||||||
|
ubuntu:) echo "$UBUNTU_DEFAULT_CODENAME" ;;
|
||||||
|
|
||||||
|
alpine:*) echo "${2:-$ALPINE_DEFAULT_VERSION}" ;;
|
||||||
|
|
||||||
|
*) return 1 ;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
# Downloads an ARM64 LXC rootfs template to $1.
|
||||||
|
# Debian: fetches latest release from community-scripts/debian-arm64-lxc on GitHub.
|
||||||
|
# Others: fetches from jenkins.linuxcontainers.org.
|
||||||
|
download_arm64_template() {
|
||||||
|
local dest="$1" url
|
||||||
|
|
||||||
|
mkdir -p "$(dirname "$dest")" || {
|
||||||
|
msg_error "Cannot create template dir."
|
||||||
|
exit 207
|
||||||
|
}
|
||||||
|
|
||||||
|
if [[ "$PCT_OSTYPE" == "debian" ]]; then
|
||||||
|
url=$(
|
||||||
|
curl -fsSL "https://api.github.com/repos/community-scripts/debian-arm64-lxc/releases/latest" |
|
||||||
|
jq -r --arg v "$CUSTOM_TEMPLATE_VARIANT" \
|
||||||
|
'.assets[].browser_download_url | select(test("debian-" + $v + "-arm64-rootfs\\.tar\\.xz$"))' |
|
||||||
|
head -n1
|
||||||
|
)
|
||||||
|
|
||||||
|
[[ -n "$url" ]] || {
|
||||||
|
msg_error "Could not find Debian ${CUSTOM_TEMPLATE_VARIANT} ARM64 template URL."
|
||||||
|
exit 207
|
||||||
|
}
|
||||||
|
else
|
||||||
|
url="https://jenkins.linuxcontainers.org/job/image-${PCT_OSTYPE}/architecture=arm64,release=${CUSTOM_TEMPLATE_VARIANT},variant=default/lastStableBuild/artifact/rootfs.tar.xz"
|
||||||
|
fi
|
||||||
|
|
||||||
|
msg_info "Downloading ${PCT_OSTYPE^} ${CUSTOM_TEMPLATE_VARIANT} ARM64 template"
|
||||||
|
if ! curl -fsSL -o "$dest" "$url"; then
|
||||||
|
msg_error "Failed to download ARM64 template from: $url"
|
||||||
|
exit 208
|
||||||
|
fi
|
||||||
|
msg_ok "Downloaded ARM64 LXC template"
|
||||||
|
}
|
||||||
|
|
||||||
|
download_template() {
|
||||||
|
local dest="${1:-$TEMPLATE_PATH}"
|
||||||
|
if [[ "$ARCH" == "arm64" ]]; then
|
||||||
|
download_arm64_template "$dest"
|
||||||
|
else
|
||||||
|
pveam download "$TEMPLATE_STORAGE" "$TEMPLATE" >>"${BUILD_LOG:-/dev/null}" 2>&1 || {
|
||||||
|
msg_error "Failed to download template '$TEMPLATE' to storage '$TEMPLATE_STORAGE'"
|
||||||
|
exit 222
|
||||||
|
}
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
# Required input variables
|
# Required input variables
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
@@ -5832,6 +5916,44 @@ create_lxc_container() {
|
|||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
# Template discovery & validation
|
# Template discovery & validation
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
|
CUSTOM_TEMPLATE_VARIANT=""
|
||||||
|
|
||||||
|
if [[ "$ARCH" == "arm64" ]]; then
|
||||||
|
# ARM64: use custom template download from linuxcontainers.org / GitHub
|
||||||
|
msg_info "Preparing ARM64 template"
|
||||||
|
|
||||||
|
CUSTOM_TEMPLATE_VARIANT=$(arm64_template_variant "$PCT_OSTYPE" "${PCT_OSVERSION:-}") || {
|
||||||
|
msg_error "No ARM64 template mapping for ${PCT_OSTYPE} ${PCT_OSVERSION:-latest}"
|
||||||
|
exit 207
|
||||||
|
}
|
||||||
|
|
||||||
|
TEMPLATE="${PCT_OSTYPE}-${CUSTOM_TEMPLATE_VARIANT}-rootfs.tar.xz"
|
||||||
|
TEMPLATE_SOURCE="custom-arm64"
|
||||||
|
|
||||||
|
# Resolve template path
|
||||||
|
TEMPLATE_PATH="$(pvesm path "${TEMPLATE_STORAGE}:vztmpl/${TEMPLATE}" 2>/dev/null || true)"
|
||||||
|
if [[ -z "$TEMPLATE_PATH" ]]; then
|
||||||
|
local _tpl_base
|
||||||
|
_tpl_base=$(awk -v s="$TEMPLATE_STORAGE" '
|
||||||
|
$0 ~ "^[^:]+:[[:space:]]*" s "$" {f=1; next}
|
||||||
|
f && /^[^[:space:]]/ {f=0}
|
||||||
|
f && $1 == "path" {print $2; exit}
|
||||||
|
' /etc/pve/storage.cfg)
|
||||||
|
TEMPLATE_PATH="${_tpl_base:-/var/lib/vz}/template/cache/$TEMPLATE"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Download if missing, too small, or corrupt
|
||||||
|
if [[ ! -f "$TEMPLATE_PATH" ]]; then
|
||||||
|
download_arm64_template "$TEMPLATE_PATH"
|
||||||
|
elif [[ "$(stat -c%s "$TEMPLATE_PATH")" -lt 1000000 ]] || ! tar -tf "$TEMPLATE_PATH" &>/dev/null; then
|
||||||
|
msg_warn "Local template invalid - re-downloading."
|
||||||
|
rm -f "$TEMPLATE_PATH"
|
||||||
|
download_arm64_template "$TEMPLATE_PATH"
|
||||||
|
else
|
||||||
|
msg_ok "Template ${BL}$TEMPLATE${CL} found locally."
|
||||||
|
fi
|
||||||
|
|
||||||
|
else
|
||||||
TEMPLATE_SEARCH="${PCT_OSTYPE}-${PCT_OSVERSION:-}"
|
TEMPLATE_SEARCH="${PCT_OSTYPE}-${PCT_OSVERSION:-}"
|
||||||
case "$PCT_OSTYPE" in
|
case "$PCT_OSTYPE" in
|
||||||
debian | ubuntu) TEMPLATE_PATTERN="-standard_" ;;
|
debian | ubuntu) TEMPLATE_PATTERN="-standard_" ;;
|
||||||
@@ -5928,7 +6050,7 @@ create_lxc_container() {
|
|||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
msg_error "No ${PCT_OSTYPE} templates available at all"
|
msg_error "No ${PCT_OSTYPE} templates available"
|
||||||
exit 225
|
exit 225
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
@@ -5964,7 +6086,7 @@ create_lxc_container() {
|
|||||||
done
|
done
|
||||||
|
|
||||||
echo ""
|
echo ""
|
||||||
read -p "Select version [1-${#AVAILABLE_VERSIONS[@]}] or Enter to exit: " choice </dev/tty
|
read -p "Select version [1-${#AVAILABLE_VERSIONS[@]}] or press Enter to exit: " choice </dev/tty
|
||||||
|
|
||||||
if [[ "$choice" =~ ^[0-9]+$ ]] && [[ "$choice" -ge 1 ]] && [[ "$choice" -le ${#AVAILABLE_VERSIONS[@]} ]]; then
|
if [[ "$choice" =~ ^[0-9]+$ ]] && [[ "$choice" -ge 1 ]] && [[ "$choice" -le ${#AVAILABLE_VERSIONS[@]} ]]; then
|
||||||
export var_version="${AVAILABLE_VERSIONS[$((choice - 1))]}"
|
export var_version="${AVAILABLE_VERSIONS[$((choice - 1))]}"
|
||||||
@@ -6037,21 +6159,21 @@ create_lxc_container() {
|
|||||||
|
|
||||||
NEED_DOWNLOAD=0
|
NEED_DOWNLOAD=0
|
||||||
if [[ ! -f "$TEMPLATE_PATH" ]]; then
|
if [[ ! -f "$TEMPLATE_PATH" ]]; then
|
||||||
msg_info "Template not present locally – will download."
|
msg_info "Template not present locally, will download it."
|
||||||
NEED_DOWNLOAD=1
|
NEED_DOWNLOAD=1
|
||||||
elif [[ ! -r "$TEMPLATE_PATH" ]]; then
|
elif [[ ! -r "$TEMPLATE_PATH" ]]; then
|
||||||
msg_error "Template file exists but is not readable – check permissions."
|
msg_error "Template file exists but is not readable, check permissions."
|
||||||
exit 221
|
exit 221
|
||||||
elif [[ "$(stat -c%s "$TEMPLATE_PATH")" -lt 1000000 ]]; then
|
elif [[ "$(stat -c%s "$TEMPLATE_PATH")" -lt 1000000 ]]; then
|
||||||
if [[ -n "$ONLINE_TEMPLATE" ]]; then
|
if [[ -n "$ONLINE_TEMPLATE" ]]; then
|
||||||
msg_warn "Template file too small (<1MB) – re-downloading."
|
msg_warn "Template file too small (<1MB), re-downloading."
|
||||||
NEED_DOWNLOAD=1
|
NEED_DOWNLOAD=1
|
||||||
else
|
else
|
||||||
msg_warn "Template looks too small, but no online version exists. Keeping local file."
|
msg_warn "Template looks too small, but no online version exists. Keeping local file."
|
||||||
fi
|
fi
|
||||||
elif ! tar -tf "$TEMPLATE_PATH" &>/dev/null; then
|
elif ! tar -tf "$TEMPLATE_PATH" &>/dev/null; then
|
||||||
if [[ -n "$ONLINE_TEMPLATE" ]]; then
|
if [[ -n "$ONLINE_TEMPLATE" ]]; then
|
||||||
msg_warn "Template appears corrupted – re-downloading."
|
msg_warn "Template appears corrupted, re-downloading."
|
||||||
NEED_DOWNLOAD=1
|
NEED_DOWNLOAD=1
|
||||||
else
|
else
|
||||||
msg_warn "Template appears corrupted, but no online version exists. Keeping local file."
|
msg_warn "Template appears corrupted, but no online version exists. Keeping local file."
|
||||||
@@ -6090,6 +6212,7 @@ create_lxc_container() {
|
|||||||
msg_error "Template $TEMPLATE not available in storage $TEMPLATE_STORAGE after download."
|
msg_error "Template $TEMPLATE not available in storage $TEMPLATE_STORAGE after download."
|
||||||
exit 223
|
exit 223
|
||||||
fi
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
# Dynamic preflight for Debian 13.x: offer upgrade if available (no hard mins)
|
# Dynamic preflight for Debian 13.x: offer upgrade if available (no hard mins)
|
||||||
@@ -6165,21 +6288,15 @@ create_lxc_container() {
|
|||||||
|
|
||||||
# Validate template before pct create (while holding lock)
|
# Validate template before pct create (while holding lock)
|
||||||
if [[ ! -s "$TEMPLATE_PATH" || "$(stat -c%s "$TEMPLATE_PATH" 2>/dev/null || echo 0)" -lt 1000000 ]]; then
|
if [[ ! -s "$TEMPLATE_PATH" || "$(stat -c%s "$TEMPLATE_PATH" 2>/dev/null || echo 0)" -lt 1000000 ]]; then
|
||||||
msg_info "Template file missing or too small – downloading"
|
msg_info "Template file missing or too small - downloading"
|
||||||
rm -f "$TEMPLATE_PATH"
|
rm -f "$TEMPLATE_PATH"
|
||||||
pveam download "$TEMPLATE_STORAGE" "$TEMPLATE" >>"${BUILD_LOG:-/dev/null}" 2>&1 || {
|
download_template
|
||||||
msg_error "Failed to download template '$TEMPLATE' to storage '$TEMPLATE_STORAGE'"
|
|
||||||
exit 222
|
|
||||||
}
|
|
||||||
msg_ok "Template downloaded"
|
msg_ok "Template downloaded"
|
||||||
elif ! tar -tf "$TEMPLATE_PATH" &>/dev/null; then
|
elif ! tar -tf "$TEMPLATE_PATH" &>/dev/null; then
|
||||||
if [[ -n "$ONLINE_TEMPLATE" ]]; then
|
if [[ "$ARCH" == "arm64" || -n "$ONLINE_TEMPLATE" ]]; then
|
||||||
msg_info "Template appears corrupted – re-downloading"
|
msg_info "Template appears corrupted - re-downloading"
|
||||||
rm -f "$TEMPLATE_PATH"
|
rm -f "$TEMPLATE_PATH"
|
||||||
pveam download "$TEMPLATE_STORAGE" "$TEMPLATE" >>"${BUILD_LOG:-/dev/null}" 2>&1 || {
|
download_template
|
||||||
msg_error "Failed to re-download template '$TEMPLATE'"
|
|
||||||
exit 222
|
|
||||||
}
|
|
||||||
msg_ok "Template re-downloaded"
|
msg_ok "Template re-downloaded"
|
||||||
else
|
else
|
||||||
msg_warn "Template appears corrupted, but no online version exists. Skipping re-download."
|
msg_warn "Template appears corrupted, but no online version exists. Skipping re-download."
|
||||||
@@ -6227,9 +6344,9 @@ create_lxc_container() {
|
|||||||
else
|
else
|
||||||
# Not a CTID collision - check if template issue and retry with fresh download
|
# Not a CTID collision - check if template issue and retry with fresh download
|
||||||
if grep -qiE 'unable to open|corrupt|invalid' "$LOGFILE"; then
|
if grep -qiE 'unable to open|corrupt|invalid' "$LOGFILE"; then
|
||||||
msg_info "Template may be corrupted – re-downloading"
|
msg_info "Template may be corrupted - re-downloading"
|
||||||
rm -f "$TEMPLATE_PATH"
|
rm -f "$TEMPLATE_PATH"
|
||||||
pveam download "$TEMPLATE_STORAGE" "$TEMPLATE" >>"${BUILD_LOG:-/dev/null}" 2>&1
|
download_template
|
||||||
msg_ok "Template re-downloaded"
|
msg_ok "Template re-downloaded"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -6242,7 +6359,11 @@ create_lxc_container() {
|
|||||||
if [[ ! -f "$LOCAL_TEMPLATE_PATH" ]]; then
|
if [[ ! -f "$LOCAL_TEMPLATE_PATH" ]]; then
|
||||||
msg_ok "Trying local storage fallback"
|
msg_ok "Trying local storage fallback"
|
||||||
msg_info "Downloading template to local"
|
msg_info "Downloading template to local"
|
||||||
|
if [[ "$ARCH" == "arm64" ]]; then
|
||||||
|
download_arm64_template "$LOCAL_TEMPLATE_PATH"
|
||||||
|
else
|
||||||
pveam download local "$TEMPLATE" >>"${BUILD_LOG:-/dev/null}" 2>&1
|
pveam download local "$TEMPLATE" >>"${BUILD_LOG:-/dev/null}" 2>&1
|
||||||
|
fi
|
||||||
msg_ok "Template downloaded to local"
|
msg_ok "Template downloaded to local"
|
||||||
else
|
else
|
||||||
msg_ok "Trying local storage fallback"
|
msg_ok "Trying local storage fallback"
|
||||||
|
|||||||
@@ -344,9 +344,15 @@ pve_check() {
|
|||||||
# - Provides link to ARM64-compatible scripts
|
# - Provides link to ARM64-compatible scripts
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
arch_check() {
|
arch_check() {
|
||||||
if [ "$(dpkg --print-architecture)" != "amd64" ]; then
|
local arch
|
||||||
msg_error "This script will not work with PiMox (ARM architecture detected)."
|
arch="$(dpkg --print-architecture)"
|
||||||
msg_warn "Visit https://github.com/asylumexp/Proxmox for ARM64 support."
|
if [[ "$arch" != "amd64" && "$arch" != "arm64" ]]; then
|
||||||
|
msg_error "This script requires amd64 or arm64 (detected: $arch)."
|
||||||
|
sleep 2
|
||||||
|
exit 106
|
||||||
|
fi
|
||||||
|
if [[ "$arch" == "arm64" && "${var_arm64:-}" != "yes" ]]; then
|
||||||
|
msg_error "This script does not yet support arm64."
|
||||||
sleep 2
|
sleep 2
|
||||||
exit 106
|
exit 106
|
||||||
fi
|
fi
|
||||||
@@ -1712,6 +1718,38 @@ function get_lxc_ip() {
|
|||||||
export LOCAL_IP
|
export LOCAL_IP
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
# ensure_whiptail()
|
||||||
|
#
|
||||||
|
# - Ensures whiptail is installed
|
||||||
|
# - Some ARM64 systems will not have whiptail installed
|
||||||
|
# - Exits with error message if installation fails
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
ensure_whiptail() {
|
||||||
|
command -v whiptail >/dev/null 2>&1 && return 0
|
||||||
|
|
||||||
|
msg_info "Installing whiptail"
|
||||||
|
apt_update_safe
|
||||||
|
$STD apt-get install -y whiptail || {
|
||||||
|
msg_error "Failed to install whiptail"
|
||||||
|
exit 100
|
||||||
|
}
|
||||||
|
msg_ok "Installed whiptail"
|
||||||
|
}
|
||||||
|
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
# arm64_notice()
|
||||||
|
#
|
||||||
|
# - Shows a short warning when running scripts on ARM64 systems
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
arm64_notice() {
|
||||||
|
[[ "$(dpkg --print-architecture)" == "arm64" ]] || return 0
|
||||||
|
|
||||||
|
whiptail --backtitle "Proxmox VE Helper Scripts" \
|
||||||
|
--title "ARM64 SUPPORT" \
|
||||||
|
--ok-button "Continue" \
|
||||||
|
--msgbox "ARM64 support is in active development.\n\nSome scripts, packages, or application releases may not be fully tested or working yet." 10 68
|
||||||
|
}
|
||||||
# ==============================================================================
|
# ==============================================================================
|
||||||
# SIGNAL TRAPS
|
# SIGNAL TRAPS
|
||||||
# ==============================================================================
|
# ==============================================================================
|
||||||
|
|||||||
@@ -99,7 +99,7 @@ if ! declare -f explain_exit_code &>/dev/null; then
|
|||||||
103) echo "Validation: Shell is not Bash" ;;
|
103) echo "Validation: Shell is not Bash" ;;
|
||||||
104) echo "Validation: Not running as root (or invoked via sudo)" ;;
|
104) echo "Validation: Not running as root (or invoked via sudo)" ;;
|
||||||
105) echo "Validation: Proxmox VE version not supported" ;;
|
105) echo "Validation: Proxmox VE version not supported" ;;
|
||||||
106) echo "Validation: Architecture not supported (ARM / PiMox)" ;;
|
106) echo "Validation: Unsupported architecture (requires amd64 or arm64)" ;;
|
||||||
107) echo "Validation: Kernel key parameters unreadable" ;;
|
107) echo "Validation: Kernel key parameters unreadable" ;;
|
||||||
108) echo "Validation: Kernel key limits exceeded" ;;
|
108) echo "Validation: Kernel key limits exceeded" ;;
|
||||||
109) echo "Proxmox: No available container ID after max attempts" ;;
|
109) echo "Proxmox: No available container ID after max attempts" ;;
|
||||||
|
|||||||
@@ -3060,10 +3060,14 @@ function fetch_and_deploy_codeberg_release() {
|
|||||||
# Fall back to architecture heuristic
|
# Fall back to architecture heuristic
|
||||||
if [[ -z "$url_match" ]]; then
|
if [[ -z "$url_match" ]]; then
|
||||||
for u in $assets; do
|
for u in $assets; do
|
||||||
if [[ "$u" =~ ($arch|amd64|x86_64|aarch64|arm64).*\.deb$ ]]; then
|
[[ "$u" =~ \.deb$ ]] || continue
|
||||||
|
if [[ "${arch,,}" =~ ^(amd64|x86_64)$ ]]; then
|
||||||
|
[[ "$u" =~ (amd64|x86_64).*\.deb$ ]] || continue
|
||||||
|
elif [[ "${arch,,}" =~ ^(arm64|aarch64)$ ]]; then
|
||||||
|
[[ "$u" =~ (arm64|aarch64).*\.deb$ ]] || continue
|
||||||
|
fi
|
||||||
url_match="$u"
|
url_match="$u"
|
||||||
break
|
break
|
||||||
fi
|
|
||||||
done
|
done
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -3360,7 +3364,11 @@ _gh_scan_older_releases() {
|
|||||||
done)
|
done)
|
||||||
fi
|
fi
|
||||||
if [[ "$has_match" != "true" ]]; then
|
if [[ "$has_match" != "true" ]]; then
|
||||||
has_match=$(echo "$releases_list" | jq -r ".[$i].assets[].browser_download_url" | grep -qE "($arch|amd64|x86_64|aarch64|arm64).*\.deb$" && echo true)
|
if [[ "${arch,,}" =~ ^(amd64|x86_64)$ ]]; then
|
||||||
|
has_match=$(echo "$releases_list" | jq -r ".[$i].assets[].browser_download_url" | grep -qE '(amd64|x86_64).*\.deb$' && echo true)
|
||||||
|
elif [[ "${arch,,}" =~ ^(arm64|aarch64)$ ]]; then
|
||||||
|
has_match=$(echo "$releases_list" | jq -r ".[$i].assets[].browser_download_url" | grep -qE '(arm64|aarch64).*\.deb$' && echo true)
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
if [[ "$has_match" != "true" ]]; then
|
if [[ "$has_match" != "true" ]]; then
|
||||||
has_match=$(echo "$releases_list" | jq -r ".[$i].assets[].browser_download_url" | grep -qE '\.deb$' && echo true)
|
has_match=$(echo "$releases_list" | jq -r ".[$i].assets[].browser_download_url" | grep -qE '\.deb$' && echo true)
|
||||||
@@ -3566,10 +3574,14 @@ function fetch_and_deploy_gh_release() {
|
|||||||
# If no match via explicit pattern, fall back to architecture heuristic
|
# If no match via explicit pattern, fall back to architecture heuristic
|
||||||
if [[ -z "$url_match" ]]; then
|
if [[ -z "$url_match" ]]; then
|
||||||
for u in $assets; do
|
for u in $assets; do
|
||||||
if [[ "$u" =~ ($arch|amd64|x86_64|aarch64|arm64).*\.deb$ ]]; then
|
[[ "$u" =~ \.deb$ ]] || continue
|
||||||
|
if [[ "${arch,,}" =~ ^(amd64|x86_64)$ ]]; then
|
||||||
|
[[ "$u" =~ (amd64|x86_64).*\.deb$ ]] || continue
|
||||||
|
elif [[ "${arch,,}" =~ ^(arm64|aarch64)$ ]]; then
|
||||||
|
[[ "$u" =~ (arm64|aarch64).*\.deb$ ]] || continue
|
||||||
|
fi
|
||||||
url_match="$u"
|
url_match="$u"
|
||||||
break
|
break
|
||||||
fi
|
|
||||||
done
|
done
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -3600,10 +3612,14 @@ function fetch_and_deploy_gh_release() {
|
|||||||
fi
|
fi
|
||||||
if [[ -z "$url_match" ]]; then
|
if [[ -z "$url_match" ]]; then
|
||||||
for u in $assets; do
|
for u in $assets; do
|
||||||
if [[ "$u" =~ ($arch|amd64|x86_64|aarch64|arm64).*\.deb$ ]]; then
|
[[ "$u" =~ \.deb$ ]] || continue
|
||||||
|
if [[ "${arch,,}" =~ ^(amd64|x86_64)$ ]]; then
|
||||||
|
[[ "$u" =~ (amd64|x86_64).*\.deb$ ]] || continue
|
||||||
|
elif [[ "${arch,,}" =~ ^(arm64|aarch64)$ ]]; then
|
||||||
|
[[ "$u" =~ (arm64|aarch64).*\.deb$ ]] || continue
|
||||||
|
fi
|
||||||
url_match="$u"
|
url_match="$u"
|
||||||
break
|
break
|
||||||
fi
|
|
||||||
done
|
done
|
||||||
fi
|
fi
|
||||||
if [[ -z "$url_match" ]]; then
|
if [[ -z "$url_match" ]]; then
|
||||||
@@ -3971,7 +3987,12 @@ function setup_ffmpeg() {
|
|||||||
|
|
||||||
# Binary fallback mode
|
# Binary fallback mode
|
||||||
if [[ "$TYPE" == "binary" ]]; then
|
if [[ "$TYPE" == "binary" ]]; then
|
||||||
if ! CURL_TIMEOUT=300 curl_with_retry "https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-amd64-static.tar.xz" "$TMP_DIR/ffmpeg.tar.xz"; then
|
local ffmpeg_arch
|
||||||
|
case "$(dpkg --print-architecture 2>/dev/null || echo amd64)" in
|
||||||
|
arm64) ffmpeg_arch="arm64" ;;
|
||||||
|
*) ffmpeg_arch="amd64" ;;
|
||||||
|
esac
|
||||||
|
if ! CURL_TIMEOUT=300 curl_with_retry "https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-${ffmpeg_arch}-static.tar.xz" "$TMP_DIR/ffmpeg.tar.xz"; then
|
||||||
msg_error "Failed to download FFmpeg binary"
|
msg_error "Failed to download FFmpeg binary"
|
||||||
rm -rf "$TMP_DIR"
|
rm -rf "$TMP_DIR"
|
||||||
return 250
|
return 250
|
||||||
@@ -4058,7 +4079,17 @@ function setup_ffmpeg() {
|
|||||||
# If no source download (either VERSION empty or download failed), use binary
|
# If no source download (either VERSION empty or download failed), use binary
|
||||||
if [[ -z "$VERSION" ]]; then
|
if [[ -z "$VERSION" ]]; then
|
||||||
msg_info "Setup FFmpeg from pre-built binary"
|
msg_info "Setup FFmpeg from pre-built binary"
|
||||||
if ! CURL_TIMEOUT=300 curl_with_retry "https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-amd64-static.tar.xz" "$TMP_DIR/ffmpeg.tar.xz"; then
|
local ffmpeg_arch detected_arch
|
||||||
|
detected_arch="$(dpkg --print-architecture 2>/dev/null || true)"
|
||||||
|
if [[ -z "$detected_arch" ]]; then
|
||||||
|
detected_arch="$(uname -m 2>/dev/null || true)"
|
||||||
|
fi
|
||||||
|
case "$detected_arch" in
|
||||||
|
arm64 | aarch64) ffmpeg_arch="arm64" ;;
|
||||||
|
amd64 | x86_64) ffmpeg_arch="amd64" ;;
|
||||||
|
*) ffmpeg_arch="amd64" ;;
|
||||||
|
esac
|
||||||
|
if ! CURL_TIMEOUT=300 curl_with_retry "https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-${ffmpeg_arch}-static.tar.xz" "$TMP_DIR/ffmpeg.tar.xz"; then
|
||||||
msg_error "Failed to download FFmpeg pre-built binary"
|
msg_error "Failed to download FFmpeg pre-built binary"
|
||||||
rm -rf "$TMP_DIR"
|
rm -rf "$TMP_DIR"
|
||||||
return 250
|
return 250
|
||||||
@@ -8234,7 +8265,19 @@ function setup_yq() {
|
|||||||
msg_info "Setup yq $LATEST_VERSION"
|
msg_info "Setup yq $LATEST_VERSION"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if ! curl_with_retry "https://github.com/${GITHUB_REPO}/releases/download/v${LATEST_VERSION}/yq_linux_amd64" "$TMP_DIR/yq"; then
|
local yq_arch detected_arch
|
||||||
|
if command -v dpkg &>/dev/null; then
|
||||||
|
detected_arch="$(dpkg --print-architecture 2>/dev/null)"
|
||||||
|
else
|
||||||
|
detected_arch="$(uname -m 2>/dev/null)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
case "$detected_arch" in
|
||||||
|
arm64 | aarch64) yq_arch="arm64" ;;
|
||||||
|
amd64 | x86_64) yq_arch="amd64" ;;
|
||||||
|
*) yq_arch="amd64" ;;
|
||||||
|
esac
|
||||||
|
if ! curl_with_retry "https://github.com/${GITHUB_REPO}/releases/download/v${LATEST_VERSION}/yq_linux_${yq_arch}" "$TMP_DIR/yq"; then
|
||||||
msg_error "Failed to download yq"
|
msg_error "Failed to download yq"
|
||||||
rm -rf "$TMP_DIR"
|
rm -rf "$TMP_DIR"
|
||||||
return 250
|
return 250
|
||||||
|
|||||||
Reference in New Issue
Block a user