| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576 |
- #!/usr/bin/env bash
- #
- # provision.sh — install the 3x-ui panel into a golden image (Packer).
- #
- # Self-contained: mirrors install.sh's download/extract logic but DELIBERATELY
- # does NOT run config_after_install and does NOT create a database. The image
- # must ship without /etc/x-ui/x-ui.db so that deploy/firstboot generates unique
- # per-instance credentials on first boot. Both x-ui.service and
- # x-ui-firstboot.service are enabled but NOT started here.
- #
- # Inputs (from Packer environment_vars):
- # XUI_VERSION release tag (e.g. v3.3.1) or 'latest'
- # XUI_ARCH amd64 (default) or arm64
- set -euo pipefail
- XUI_VERSION="${XUI_VERSION:-latest}"
- XUI_ARCH="${XUI_ARCH:-amd64}"
- XUI_DIR="/usr/local/x-ui"
- REPO="MHSanaei/3x-ui"
- export DEBIAN_FRONTEND=noninteractive
- echo "[provision] installing base packages..."
- apt-get update
- apt-get install -y --no-install-recommends \
- ca-certificates curl tar tzdata socat openssl cron jq
- echo "[provision] resolving 3x-ui version..."
- if [ "$XUI_VERSION" = "latest" ]; then
- XUI_VERSION=$(curl -fsSL "https://api.github.com/repos/${REPO}/releases/latest" | jq -r '.tag_name')
- fi
- if [ -z "$XUI_VERSION" ] || [ "$XUI_VERSION" = "null" ]; then
- echo "[provision] ERROR: could not resolve 3x-ui release tag" >&2
- exit 1
- fi
- echo "[provision] installing 3x-ui ${XUI_VERSION} (${XUI_ARCH})"
- tarball="x-ui-linux-${XUI_ARCH}.tar.gz"
- url="https://github.com/${REPO}/releases/download/${XUI_VERSION}/${tarball}"
- tmp="$(mktemp -d)"
- trap 'rm -rf "$tmp"' EXIT
- # Download the RELEASED binary tarball (no Go build inside the image).
- curl -fL4 --retry 3 -o "${tmp}/${tarball}" "$url"
- # Extract into /usr/local/ (the tarball contains an x-ui/ directory).
- systemctl stop x-ui > /dev/null 2>&1 || true
- rm -rf "$XUI_DIR"
- tar -xzf "${tmp}/${tarball}" -C /usr/local/
- chmod +x "${XUI_DIR}/x-ui" "${XUI_DIR}/x-ui.sh"
- chmod +x "${XUI_DIR}"/bin/* 2> /dev/null || true
- # Install the x-ui management CLI.
- if [ -f "${XUI_DIR}/x-ui.sh" ]; then
- cp -f "${XUI_DIR}/x-ui.sh" /usr/bin/x-ui
- else
- curl -fL4 -o /usr/bin/x-ui "https://raw.githubusercontent.com/${REPO}/main/x-ui.sh"
- fi
- chmod +x /usr/bin/x-ui
- mkdir -p /var/log/x-ui
- # Panel systemd unit (Ubuntu base => debian variant).
- install -m 644 "${XUI_DIR}/x-ui.service.debian" /etc/systemd/system/x-ui.service
- # First-boot per-instance credential unit + script (uploaded to /tmp/firstboot).
- install -m 755 /tmp/firstboot/x-ui-firstboot.sh "${XUI_DIR}/x-ui-firstboot.sh"
- install -m 644 /tmp/firstboot/x-ui-firstboot.service /etc/systemd/system/x-ui-firstboot.service
- systemctl daemon-reload
- # Enable (start on next boot) but do NOT start now — there is no DB yet.
- systemctl enable x-ui-firstboot.service
- systemctl enable x-ui.service
- # Belt-and-braces: ensure no DB / sentinel was created during provisioning.
- rm -f /etc/x-ui/x-ui.db /etc/x-ui/x-ui.db-* /etc/x-ui/.firstboot-done 2> /dev/null || true
- echo "[provision] done — panel installed, services enabled, NO database initialized."
|