2025-05-05 15:47:43 +02:00
|
|
|
|
#!/bin/bash
|
2025-05-15 15:18:41 +02:00
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs Node.js and optional global modules.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Installs specified Node.js version using NodeSource APT repo
|
|
|
|
|
|
# - Optionally installs or updates global npm modules
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# NODE_VERSION - Node.js version to install (default: 22)
|
|
|
|
|
|
# NODE_MODULE - Comma-separated list of global modules (e.g. "yarn,@vue/cli@5.0.0")
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_nodejs() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local NODE_VERSION="${NODE_VERSION:-22}"
|
|
|
|
|
|
local NODE_MODULE="${NODE_MODULE:-}"
|
|
|
|
|
|
local CURRENT_NODE_VERSION=""
|
|
|
|
|
|
local NEED_NODE_INSTALL=false
|
|
|
|
|
|
|
|
|
|
|
|
# Check if Node.js is already installed
|
|
|
|
|
|
if command -v node >/dev/null; then
|
|
|
|
|
|
CURRENT_NODE_VERSION="$(node -v | grep -oP '^v\K[0-9]+')"
|
|
|
|
|
|
if [[ "$CURRENT_NODE_VERSION" != "$NODE_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Old Node.js $CURRENT_NODE_VERSION found, replacing with $NODE_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
NEED_NODE_INSTALL=true
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Node.js $NODE_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
NEED_NODE_INSTALL=true
|
|
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-27 13:48:34 +02:00
|
|
|
|
if ! command -v jq &>/dev/null; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y jq || {
|
2025-05-27 13:48:34 +02:00
|
|
|
|
msg_error "Failed to install jq"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# Install Node.js if required
|
|
|
|
|
|
if [[ "$NEED_NODE_INSTALL" == true ]]; then
|
|
|
|
|
|
$STD apt-get purge -y nodejs
|
|
|
|
|
|
rm -f /etc/apt/sources.list.d/nodesource.list /etc/apt/keyrings/nodesource.gpg
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
mkdir -p /etc/apt/keyrings
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if ! curl -fsSL https://deb.nodesource.com/gpgkey/nodesource-repo.gpg.key |
|
|
|
|
|
|
gpg --dearmor -o /etc/apt/keyrings/nodesource.gpg; then
|
|
|
|
|
|
msg_error "Failed to download or import NodeSource GPG key"
|
|
|
|
|
|
exit 1
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
echo "deb [signed-by=/etc/apt/keyrings/nodesource.gpg] https://deb.nodesource.com/node_${NODE_VERSION}.x nodistro main" \
|
|
|
|
|
|
>/etc/apt/sources.list.d/nodesource.list
|
|
|
|
|
|
|
2025-07-10 13:58:20 +02:00
|
|
|
|
sleep 2
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if ! apt-get update >/dev/null 2>&1; then
|
2025-07-10 13:58:20 +02:00
|
|
|
|
msg_warn "APT update failed – retrying in 5s"
|
|
|
|
|
|
sleep 5
|
|
|
|
|
|
if ! apt-get update >/dev/null 2>&1; then
|
|
|
|
|
|
msg_error "Failed to update APT repositories after adding NodeSource"
|
|
|
|
|
|
exit 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if ! apt-get install -y nodejs >/dev/null 2>&1; then
|
|
|
|
|
|
msg_error "Failed to install Node.js ${NODE_VERSION} from NodeSource"
|
|
|
|
|
|
exit 1
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-26 20:49:17 +02:00
|
|
|
|
# Update to latest npm
|
|
|
|
|
|
$STD npm install -g npm@latest || {
|
|
|
|
|
|
msg_error "Failed to update npm to latest version"
|
|
|
|
|
|
}
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Node.js ${NODE_VERSION}"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
export NODE_OPTIONS="--max-old-space-size=4096"
|
2025-06-26 19:04:14 +02:00
|
|
|
|
|
|
|
|
|
|
# Ensure valid working directory for npm (avoids uv_cwd error)
|
2025-06-26 18:24:43 +02:00
|
|
|
|
if [[ ! -d /opt ]]; then
|
|
|
|
|
|
mkdir -p /opt
|
|
|
|
|
|
fi
|
2025-06-26 19:04:14 +02:00
|
|
|
|
cd /opt || {
|
2025-06-26 18:24:43 +02:00
|
|
|
|
msg_error "Failed to set safe working directory before npm install"
|
|
|
|
|
|
exit 1
|
|
|
|
|
|
}
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# Install global Node modules
|
|
|
|
|
|
if [[ -n "$NODE_MODULE" ]]; then
|
|
|
|
|
|
IFS=',' read -ra MODULES <<<"$NODE_MODULE"
|
|
|
|
|
|
for mod in "${MODULES[@]}"; do
|
|
|
|
|
|
local MODULE_NAME MODULE_REQ_VERSION MODULE_INSTALLED_VERSION
|
2025-05-13 10:38:19 +02:00
|
|
|
|
if [[ "$mod" == @*/*@* ]]; then
|
|
|
|
|
|
# Scoped package with version, e.g. @vue/cli-service@latest
|
2025-05-05 15:47:43 +02:00
|
|
|
|
MODULE_NAME="${mod%@*}"
|
2025-05-13 10:38:19 +02:00
|
|
|
|
MODULE_REQ_VERSION="${mod##*@}"
|
|
|
|
|
|
elif [[ "$mod" == *"@"* ]]; then
|
|
|
|
|
|
# Unscoped package with version, e.g. yarn@latest
|
|
|
|
|
|
MODULE_NAME="${mod%@*}"
|
|
|
|
|
|
MODULE_REQ_VERSION="${mod##*@}"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
2025-05-13 10:38:19 +02:00
|
|
|
|
# No version specified
|
2025-05-05 15:47:43 +02:00
|
|
|
|
MODULE_NAME="$mod"
|
|
|
|
|
|
MODULE_REQ_VERSION="latest"
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
# Check if the module is already installed
|
|
|
|
|
|
if npm list -g --depth=0 "$MODULE_NAME" >/dev/null 2>&1; then
|
|
|
|
|
|
MODULE_INSTALLED_VERSION="$(npm list -g --depth=0 "$MODULE_NAME" | grep "$MODULE_NAME@" | awk -F@ '{print $2}' | tr -d '[:space:]')"
|
|
|
|
|
|
if [[ "$MODULE_REQ_VERSION" != "latest" && "$MODULE_REQ_VERSION" != "$MODULE_INSTALLED_VERSION" ]]; then
|
|
|
|
|
|
msg_info "Updating $MODULE_NAME from v$MODULE_INSTALLED_VERSION to v$MODULE_REQ_VERSION"
|
|
|
|
|
|
if ! $STD npm install -g "${MODULE_NAME}@${MODULE_REQ_VERSION}"; then
|
|
|
|
|
|
msg_error "Failed to update $MODULE_NAME to version $MODULE_REQ_VERSION"
|
|
|
|
|
|
exit 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
elif [[ "$MODULE_REQ_VERSION" == "latest" ]]; then
|
|
|
|
|
|
msg_info "Updating $MODULE_NAME to latest version"
|
|
|
|
|
|
if ! $STD npm install -g "${MODULE_NAME}@latest"; then
|
|
|
|
|
|
msg_error "Failed to update $MODULE_NAME to latest version"
|
|
|
|
|
|
exit 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
else
|
|
|
|
|
|
msg_info "Installing $MODULE_NAME@$MODULE_REQ_VERSION"
|
|
|
|
|
|
if ! $STD npm install -g "${MODULE_NAME}@${MODULE_REQ_VERSION}"; then
|
|
|
|
|
|
msg_error "Failed to install $MODULE_NAME@$MODULE_REQ_VERSION"
|
|
|
|
|
|
exit 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
done
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Installed Node.js modules: $NODE_MODULE"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
}
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
2025-05-27 14:29:50 +02:00
|
|
|
|
# Installs or upgrades PostgreSQL and optional extensions/modules.
|
2025-05-15 15:18:41 +02:00
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Detects existing PostgreSQL version
|
|
|
|
|
|
# - Dumps all databases before upgrade
|
|
|
|
|
|
# - Adds PGDG repo and installs specified version
|
2025-05-27 14:29:50 +02:00
|
|
|
|
# - Installs optional PG_MODULES (e.g. postgis, contrib)
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# - Restores dumped data post-upgrade
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# PG_VERSION - Major PostgreSQL version (e.g. 15, 16) (default: 16)
|
2025-05-27 14:29:50 +02:00
|
|
|
|
# PG_MODULES - Comma-separated list of extensions (e.g. "postgis,contrib")
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_postgresql() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local PG_VERSION="${PG_VERSION:-16}"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
local PG_MODULES="${PG_MODULES:-}"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local CURRENT_PG_VERSION=""
|
|
|
|
|
|
local DISTRO
|
|
|
|
|
|
local NEED_PG_INSTALL=false
|
|
|
|
|
|
DISTRO="$(awk -F'=' '/^VERSION_CODENAME=/{ print $NF }' /etc/os-release)"
|
|
|
|
|
|
|
|
|
|
|
|
if command -v psql >/dev/null; then
|
2025-05-13 10:38:19 +02:00
|
|
|
|
CURRENT_PG_VERSION="$(psql -V | awk '{print $3}' | cut -d. -f1)"
|
|
|
|
|
|
if [[ "$CURRENT_PG_VERSION" == "$PG_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
: # PostgreSQL is already at the desired version – no action needed
|
2025-05-27 14:29:50 +02:00
|
|
|
|
else
|
|
|
|
|
|
msg_info "Detected PostgreSQL $CURRENT_PG_VERSION, preparing upgrade to $PG_VERSION"
|
|
|
|
|
|
NEED_PG_INSTALL=true
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
else
|
2025-05-15 15:18:41 +02:00
|
|
|
|
msg_info "Setup PostgreSQL $PG_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
NEED_PG_INSTALL=true
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$NEED_PG_INSTALL" == true ]]; then
|
2025-05-13 10:38:19 +02:00
|
|
|
|
if [[ -n "$CURRENT_PG_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Dumping PostgreSQL $CURRENT_PG_VERSION data"
|
2025-05-13 10:38:19 +02:00
|
|
|
|
su - postgres -c "pg_dumpall > /var/lib/postgresql/backup_$(date +%F)_v${CURRENT_PG_VERSION}.sql"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Data dump completed"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
systemctl stop postgresql
|
|
|
|
|
|
fi
|
2025-05-13 10:38:19 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
rm -f /etc/apt/sources.list.d/pgdg.list /etc/apt/trusted.gpg.d/postgresql.gpg
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Adding PostgreSQL PGDG repository"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
curl -fsSL https://www.postgresql.org/media/keys/ACCC4CF8.asc |
|
|
|
|
|
|
gpg --dearmor -o /etc/apt/trusted.gpg.d/postgresql.gpg
|
|
|
|
|
|
|
|
|
|
|
|
echo "deb https://apt.postgresql.org/pub/repos/apt ${DISTRO}-pgdg main" \
|
|
|
|
|
|
>/etc/apt/sources.list.d/pgdg.list
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_ok "Repository added"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get update
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup PostgreSQL $PG_VERSION"
|
2025-05-13 10:38:19 +02:00
|
|
|
|
$STD apt-get install -y "postgresql-${PG_VERSION}" "postgresql-client-${PG_VERSION}"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup PostgreSQL $PG_VERSION"
|
2025-05-13 10:38:19 +02:00
|
|
|
|
|
|
|
|
|
|
if [[ -n "$CURRENT_PG_VERSION" ]]; then
|
|
|
|
|
|
$STD apt-get purge -y "postgresql-${CURRENT_PG_VERSION}" "postgresql-client-${CURRENT_PG_VERSION}" || true
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Starting PostgreSQL $PG_VERSION"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
systemctl enable -q --now postgresql
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_ok "PostgreSQL $PG_VERSION started"
|
2025-05-13 10:38:19 +02:00
|
|
|
|
|
|
|
|
|
|
if [[ -n "$CURRENT_PG_VERSION" ]]; then
|
2025-05-27 14:29:50 +02:00
|
|
|
|
msg_info "Restoring dumped data"
|
2025-05-13 10:38:19 +02:00
|
|
|
|
su - postgres -c "psql < /var/lib/postgresql/backup_$(date +%F)_v${CURRENT_PG_VERSION}.sql"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Data restored"
|
2025-05-13 10:38:19 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
msg_ok "PostgreSQL $PG_VERSION installed"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-05-27 14:29:50 +02:00
|
|
|
|
|
|
|
|
|
|
# Install optional PostgreSQL modules
|
|
|
|
|
|
if [[ -n "$PG_MODULES" ]]; then
|
|
|
|
|
|
IFS=',' read -ra MODULES <<<"$PG_MODULES"
|
|
|
|
|
|
for module in "${MODULES[@]}"; do
|
|
|
|
|
|
local pkg="postgresql-${PG_VERSION}-${module}"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup PostgreSQL module/s: $pkg"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
$STD apt-get install -y "$pkg" || {
|
|
|
|
|
|
msg_error "Failed to install $pkg"
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
done
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup PostgreSQL modules"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs or updates MariaDB from official repo.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Detects current MariaDB version and replaces it if necessary
|
|
|
|
|
|
# - Preserves existing database data
|
|
|
|
|
|
# - Dynamically determines latest GA version if "latest" is given
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# MARIADB_VERSION - MariaDB version to install (e.g. 10.11, latest) (default: latest)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
setup_mariadb() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local MARIADB_VERSION="${MARIADB_VERSION:-latest}"
|
|
|
|
|
|
local DISTRO_CODENAME
|
|
|
|
|
|
DISTRO_CODENAME="$(awk -F= '/^VERSION_CODENAME=/{print $2}' /etc/os-release)"
|
2025-06-14 23:13:59 +02:00
|
|
|
|
CURRENT_OS="$(awk -F= '/^ID=/{print $2}' /etc/os-release)"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
2025-07-04 21:22:50 +02:00
|
|
|
|
if ! curl -fsI http://mirror.mariadb.org/repo/ >/dev/null; then
|
|
|
|
|
|
msg_error "MariaDB mirror not reachable"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setting up MariaDB $MARIADB_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# grab dynamic latest LTS version
|
|
|
|
|
|
if [[ "$MARIADB_VERSION" == "latest" ]]; then
|
2025-05-15 15:18:41 +02:00
|
|
|
|
MARIADB_VERSION=$(curl -fsSL http://mirror.mariadb.org/repo/ |
|
|
|
|
|
|
grep -Eo '[0-9]+\.[0-9]+\.[0-9]+/' |
|
|
|
|
|
|
grep -vE 'rc/|rolling/' |
|
|
|
|
|
|
sed 's|/||' |
|
|
|
|
|
|
sort -Vr |
|
|
|
|
|
|
head -n1)
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if [[ -z "$MARIADB_VERSION" ]]; then
|
2025-05-15 15:18:41 +02:00
|
|
|
|
msg_error "Could not determine latest GA MariaDB version"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
local CURRENT_VERSION=""
|
|
|
|
|
|
if command -v mariadb >/dev/null; then
|
2025-05-28 13:17:28 +02:00
|
|
|
|
CURRENT_VERSION=$(mariadb --version | grep -oE '[0-9]+\.[0-9]+\.[0-9]+')
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$CURRENT_VERSION" == "$MARIADB_VERSION" ]]; then
|
2025-05-15 15:18:41 +02:00
|
|
|
|
$STD msg_info "MariaDB $MARIADB_VERSION, upgrading"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
$STD apt-get update
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get install --only-upgrade -y mariadb-server mariadb-client
|
2025-05-15 15:18:41 +02:00
|
|
|
|
$STD msg_ok "MariaDB upgraded to $MARIADB_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
return 0
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -n "$CURRENT_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Upgrading MariaDB $CURRENT_VERSION to $MARIADB_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD systemctl stop mariadb >/dev/null 2>&1 || true
|
|
|
|
|
|
$STD apt-get purge -y 'mariadb*' || true
|
|
|
|
|
|
rm -f /etc/apt/sources.list.d/mariadb.list /etc/apt/trusted.gpg.d/mariadb.gpg
|
|
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Setup MariaDB $MARIADB_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
curl -fsSL "https://mariadb.org/mariadb_release_signing_key.asc" |
|
|
|
|
|
|
gpg --dearmor -o /etc/apt/trusted.gpg.d/mariadb.gpg
|
|
|
|
|
|
|
2025-06-14 23:13:59 +02:00
|
|
|
|
echo "deb [signed-by=/etc/apt/trusted.gpg.d/mariadb.gpg] http://mirror.mariadb.org/repo/${MARIADB_VERSION}/${CURRENT_OS} ${DISTRO_CODENAME} main" \
|
2025-05-05 15:47:43 +02:00
|
|
|
|
>/etc/apt/sources.list.d/mariadb.list
|
|
|
|
|
|
|
|
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y mariadb-server mariadb-client
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
msg_ok "Setup MariaDB $MARIADB_VERSION"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs or upgrades MySQL and configures APT repo.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Detects existing MySQL installation
|
|
|
|
|
|
# - Purges conflicting packages before installation
|
|
|
|
|
|
# - Supports clean upgrade
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# MYSQL_VERSION - MySQL version to install (e.g. 5.7, 8.0) (default: 8.0)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_mysql() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local MYSQL_VERSION="${MYSQL_VERSION:-8.0}"
|
|
|
|
|
|
local CURRENT_VERSION=""
|
|
|
|
|
|
local NEED_INSTALL=false
|
2025-06-18 12:07:08 +02:00
|
|
|
|
CURRENT_OS="$(awk -F= '/^ID=/{print $2}' /etc/os-release)"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
|
|
|
|
|
if command -v mysql >/dev/null; then
|
|
|
|
|
|
CURRENT_VERSION="$(mysql --version | grep -oP 'Distrib\s+\K[0-9]+\.[0-9]+')"
|
|
|
|
|
|
if [[ "$CURRENT_VERSION" != "$MYSQL_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "MySQL $CURRENT_VERSION will be upgraded to $MYSQL_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
NEED_INSTALL=true
|
2025-04-28 15:25:10 +02:00
|
|
|
|
else
|
2025-06-02 11:41:25 +02:00
|
|
|
|
# Check for patch-level updates
|
|
|
|
|
|
if apt list --upgradable 2>/dev/null | grep -q '^mysql-server/'; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "MySQL $CURRENT_VERSION available for upgrade"
|
2025-06-02 11:41:25 +02:00
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install --only-upgrade -y mysql-server
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_ok "MySQL upgraded"
|
2025-06-02 11:41:25 +02:00
|
|
|
|
fi
|
|
|
|
|
|
return
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup MySQL $MYSQL_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
NEED_INSTALL=true
|
|
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if [[ "$NEED_INSTALL" == true ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD systemctl stop mysql || true
|
2025-06-02 11:41:25 +02:00
|
|
|
|
$STD apt-get purge -y "^mysql-server.*" "^mysql-client.*" "^mysql-common.*" || true
|
2025-05-05 15:47:43 +02:00
|
|
|
|
rm -f /etc/apt/sources.list.d/mysql.list /etc/apt/trusted.gpg.d/mysql.gpg
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-06-02 11:41:25 +02:00
|
|
|
|
local DISTRO_CODENAME
|
2025-05-05 15:47:43 +02:00
|
|
|
|
DISTRO_CODENAME="$(awk -F= '/VERSION_CODENAME/ { print $2 }' /etc/os-release)"
|
2025-06-02 11:41:25 +02:00
|
|
|
|
curl -fsSL https://repo.mysql.com/RPM-GPG-KEY-mysql-2023 | gpg --dearmor -o /etc/apt/trusted.gpg.d/mysql.gpg
|
2025-06-18 12:07:08 +02:00
|
|
|
|
echo "deb [signed-by=/etc/apt/trusted.gpg.d/mysql.gpg] https://repo.mysql.com/apt/${CURRENT_OS}/ ${DISTRO_CODENAME} mysql-${MYSQL_VERSION}" \
|
2025-05-05 15:47:43 +02:00
|
|
|
|
>/etc/apt/sources.list.d/mysql.list
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-06-02 11:41:25 +02:00
|
|
|
|
export DEBIAN_FRONTEND=noninteractive
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y mysql-server
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup MySQL $MYSQL_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs PHP with selected modules and configures Apache/FPM support.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Adds Sury PHP repo if needed
|
|
|
|
|
|
# - Installs default and user-defined modules
|
|
|
|
|
|
# - Patches php.ini for CLI, Apache, and FPM as needed
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# PHP_VERSION - PHP version to install (default: 8.4)
|
|
|
|
|
|
# PHP_MODULE - Additional comma-separated modules
|
|
|
|
|
|
# PHP_APACHE - Set YES to enable PHP with Apache
|
|
|
|
|
|
# PHP_FPM - Set YES to enable PHP-FPM
|
|
|
|
|
|
# PHP_MEMORY_LIMIT - (default: 512M)
|
|
|
|
|
|
# PHP_UPLOAD_MAX_FILESIZE - (default: 128M)
|
|
|
|
|
|
# PHP_POST_MAX_SIZE - (default: 128M)
|
|
|
|
|
|
# PHP_MAX_EXECUTION_TIME - (default: 300)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_php() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local PHP_VERSION="${PHP_VERSION:-8.4}"
|
|
|
|
|
|
local PHP_MODULE="${PHP_MODULE:-}"
|
|
|
|
|
|
local PHP_APACHE="${PHP_APACHE:-NO}"
|
|
|
|
|
|
local PHP_FPM="${PHP_FPM:-NO}"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
local DISTRO_CODENAME
|
|
|
|
|
|
DISTRO_CODENAME=$(awk -F= '/VERSION_CODENAME/ { print $2 }' /etc/os-release)
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local DEFAULT_MODULES="bcmath,cli,curl,gd,intl,mbstring,opcache,readline,xml,zip"
|
|
|
|
|
|
local COMBINED_MODULES
|
|
|
|
|
|
|
|
|
|
|
|
local PHP_MEMORY_LIMIT="${PHP_MEMORY_LIMIT:-512M}"
|
|
|
|
|
|
local PHP_UPLOAD_MAX_FILESIZE="${PHP_UPLOAD_MAX_FILESIZE:-128M}"
|
|
|
|
|
|
local PHP_POST_MAX_SIZE="${PHP_POST_MAX_SIZE:-128M}"
|
|
|
|
|
|
local PHP_MAX_EXECUTION_TIME="${PHP_MAX_EXECUTION_TIME:-300}"
|
|
|
|
|
|
|
|
|
|
|
|
# Merge default + user-defined modules
|
|
|
|
|
|
if [[ -n "$PHP_MODULE" ]]; then
|
|
|
|
|
|
COMBINED_MODULES="${DEFAULT_MODULES},${PHP_MODULE}"
|
|
|
|
|
|
else
|
|
|
|
|
|
COMBINED_MODULES="${DEFAULT_MODULES}"
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-07-22 13:14:09 +02:00
|
|
|
|
# Deduplicate
|
2025-05-05 15:47:43 +02:00
|
|
|
|
COMBINED_MODULES=$(echo "$COMBINED_MODULES" | tr ',' '\n' | awk '!seen[$0]++' | paste -sd, -)
|
|
|
|
|
|
|
2025-07-22 13:14:09 +02:00
|
|
|
|
# Get current PHP-CLI version
|
2025-06-16 14:20:46 +02:00
|
|
|
|
local CURRENT_PHP=""
|
2025-05-13 10:38:19 +02:00
|
|
|
|
if command -v php >/dev/null 2>&1; then
|
|
|
|
|
|
CURRENT_PHP=$(php -v 2>/dev/null | awk '/^PHP/{print $2}' | cut -d. -f1,2)
|
|
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
2025-05-27 14:29:50 +02:00
|
|
|
|
if [[ -z "$CURRENT_PHP" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup PHP $PHP_VERSION"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
elif [[ "$CURRENT_PHP" != "$PHP_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Old PHP $CURRENT_PHP detected, Setup new PHP $PHP_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get purge -y "php${CURRENT_PHP//./}"* || true
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-16 14:20:46 +02:00
|
|
|
|
# Ensure Sury repo is available
|
|
|
|
|
|
if [[ ! -f /etc/apt/sources.list.d/php.list ]]; then
|
|
|
|
|
|
$STD curl -fsSLo /tmp/debsuryorg-archive-keyring.deb https://packages.sury.org/debsuryorg-archive-keyring.deb
|
|
|
|
|
|
$STD dpkg -i /tmp/debsuryorg-archive-keyring.deb
|
|
|
|
|
|
echo "deb [signed-by=/usr/share/keyrings/deb.sury.org-php.gpg] https://packages.sury.org/php/ ${DISTRO_CODENAME} main" \
|
|
|
|
|
|
>/etc/apt/sources.list.d/php.list
|
|
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-07-22 13:14:09 +02:00
|
|
|
|
# Build module list
|
2025-07-07 22:48:56 +02:00
|
|
|
|
local MODULE_LIST="php${PHP_VERSION}"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
IFS=',' read -ra MODULES <<<"$COMBINED_MODULES"
|
|
|
|
|
|
for mod in "${MODULES[@]}"; do
|
2025-07-22 13:14:09 +02:00
|
|
|
|
if apt-cache show "php${PHP_VERSION}-${mod}" >/dev/null 2>&1; then
|
|
|
|
|
|
MODULE_LIST+=" php${PHP_VERSION}-${mod}"
|
|
|
|
|
|
else
|
|
|
|
|
|
msg_warn "PHP-Module ${mod} for PHP ${PHP_VERSION} not found – skipping"
|
|
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
done
|
2025-05-27 14:29:50 +02:00
|
|
|
|
if [[ "$PHP_FPM" == "YES" ]]; then
|
|
|
|
|
|
MODULE_LIST+=" php${PHP_VERSION}-fpm"
|
|
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
2025-07-22 13:14:09 +02:00
|
|
|
|
# install apache2 with PHP support if requested
|
|
|
|
|
|
if [[ "$PHP_APACHE" == "YES" ]]; then
|
|
|
|
|
|
if ! dpkg -l | grep -q "libapache2-mod-php${PHP_VERSION}"; then
|
|
|
|
|
|
$STD msg_info "Installing Apache with PHP${PHP_VERSION} support"
|
|
|
|
|
|
$STD apt-get install -y apache2 libapache2-mod-php"${PHP_VERSION}"
|
|
|
|
|
|
$STD msg_ok "Setup Apache with PHP${PHP_VERSION}"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-07-22 13:14:09 +02:00
|
|
|
|
# setup / update PHP modules
|
|
|
|
|
|
$STD apt-get install -y "$MODULE_LIST"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup PHP $PHP_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
2025-07-22 13:14:09 +02:00
|
|
|
|
# optional stop old PHP-FPM service
|
|
|
|
|
|
if [[ "$PHP_FPM" == "YES" && -n "$CURRENT_PHP" && "$CURRENT_PHP" != "$PHP_VERSION" ]]; then
|
|
|
|
|
|
$STD systemctl stop php"${CURRENT_PHP}"-fpm || true
|
|
|
|
|
|
$STD systemctl disable php"${CURRENT_PHP}"-fpm || true
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
# Patch all relevant php.ini files
|
2025-06-16 14:20:46 +02:00
|
|
|
|
local PHP_INI_PATHS=("/etc/php/${PHP_VERSION}/cli/php.ini")
|
2025-05-05 15:47:43 +02:00
|
|
|
|
[[ "$PHP_FPM" == "YES" ]] && PHP_INI_PATHS+=("/etc/php/${PHP_VERSION}/fpm/php.ini")
|
|
|
|
|
|
[[ "$PHP_APACHE" == "YES" ]] && PHP_INI_PATHS+=("/etc/php/${PHP_VERSION}/apache2/php.ini")
|
|
|
|
|
|
for ini in "${PHP_INI_PATHS[@]}"; do
|
|
|
|
|
|
if [[ -f "$ini" ]]; then
|
2025-05-15 15:18:41 +02:00
|
|
|
|
$STD msg_info "Patching $ini"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
sed -i "s|^memory_limit = .*|memory_limit = ${PHP_MEMORY_LIMIT}|" "$ini"
|
|
|
|
|
|
sed -i "s|^upload_max_filesize = .*|upload_max_filesize = ${PHP_UPLOAD_MAX_FILESIZE}|" "$ini"
|
|
|
|
|
|
sed -i "s|^post_max_size = .*|post_max_size = ${PHP_POST_MAX_SIZE}|" "$ini"
|
|
|
|
|
|
sed -i "s|^max_execution_time = .*|max_execution_time = ${PHP_MAX_EXECUTION_TIME}|" "$ini"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
$STD msg_ok "Patched $ini"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
done
|
2025-07-22 13:14:09 +02:00
|
|
|
|
|
|
|
|
|
|
# patch Apache configuration if needed
|
|
|
|
|
|
if [[ "$PHP_APACHE" == "YES" ]]; then
|
|
|
|
|
|
for mod in $(ls /etc/apache2/mods-enabled/ 2>/dev/null | grep -E '^php[0-9]\.[0-9]\.conf$' | sed 's/\.conf//'); do
|
|
|
|
|
|
if [[ "$mod" != "php${PHP_VERSION}" ]]; then
|
|
|
|
|
|
$STD a2dismod "$mod" || true
|
|
|
|
|
|
fi
|
|
|
|
|
|
done
|
|
|
|
|
|
$STD a2enmod mpm_prefork
|
|
|
|
|
|
$STD a2enmod "php${PHP_VERSION}"
|
|
|
|
|
|
$STD systemctl restart apache2 || true
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
# enable and restart PHP-FPM if requested
|
|
|
|
|
|
if [[ "$PHP_FPM" == "YES" ]]; then
|
|
|
|
|
|
if systemctl list-unit-files | grep -q "php${PHP_VERSION}-fpm.service"; then
|
|
|
|
|
|
$STD systemctl enable php"${PHP_VERSION}"-fpm
|
|
|
|
|
|
$STD systemctl restart php"${PHP_VERSION}"-fpm
|
|
|
|
|
|
else
|
|
|
|
|
|
msg_warn "FPM requested but service php${PHP_VERSION}-fpm not found"
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs or updates Composer globally.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Downloads latest version from getcomposer.org
|
|
|
|
|
|
# - Installs to /usr/local/bin/composer
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_composer() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local COMPOSER_BIN="/usr/local/bin/composer"
|
|
|
|
|
|
export COMPOSER_ALLOW_SUPERUSER=1
|
|
|
|
|
|
|
|
|
|
|
|
# Check if composer is already installed
|
|
|
|
|
|
if [[ -x "$COMPOSER_BIN" ]]; then
|
|
|
|
|
|
local CURRENT_VERSION
|
|
|
|
|
|
CURRENT_VERSION=$("$COMPOSER_BIN" --version | awk '{print $3}')
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Old Composer $CURRENT_VERSION found, updating to latest"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
2025-05-15 15:18:41 +02:00
|
|
|
|
msg_info "Setup Composer"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
# Download and install latest composer
|
|
|
|
|
|
curl -fsSL https://getcomposer.org/installer -o /tmp/composer-setup.php
|
2025-05-27 14:29:50 +02:00
|
|
|
|
php /tmp/composer-setup.php --install-dir=/usr/local/bin --filename=composer >/dev/null 2>&1
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
|
|
|
|
|
if [[ $? -ne 0 ]]; then
|
|
|
|
|
|
msg_error "Failed to install Composer"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
chmod +x "$COMPOSER_BIN"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
composer diagnose >/dev/null 2>&1
|
|
|
|
|
|
msg_ok "Setup Composer"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs Go (Golang) from official tarball.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Determines system architecture
|
|
|
|
|
|
# - Downloads latest version if GO_VERSION not set
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# GO_VERSION - Version to install (e.g. 1.22.2 or latest)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_go() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local ARCH
|
|
|
|
|
|
case "$(uname -m)" in
|
|
|
|
|
|
x86_64) ARCH="amd64" ;;
|
|
|
|
|
|
aarch64) ARCH="arm64" ;;
|
|
|
|
|
|
*)
|
|
|
|
|
|
msg_error "Unsupported architecture: $(uname -m)"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
;;
|
|
|
|
|
|
esac
|
|
|
|
|
|
|
|
|
|
|
|
# Determine version
|
|
|
|
|
|
if [[ -z "${GO_VERSION:-}" || "${GO_VERSION}" == "latest" ]]; then
|
|
|
|
|
|
GO_VERSION=$(curl -fsSL https://go.dev/VERSION?m=text | head -n1 | sed 's/^go//')
|
|
|
|
|
|
if [[ -z "$GO_VERSION" ]]; then
|
|
|
|
|
|
msg_error "Could not determine latest Go version"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
local GO_BIN="/usr/local/bin/go"
|
|
|
|
|
|
local GO_INSTALL_DIR="/usr/local/go"
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -x "$GO_BIN" ]]; then
|
|
|
|
|
|
local CURRENT_VERSION
|
|
|
|
|
|
CURRENT_VERSION=$("$GO_BIN" version | awk '{print $3}' | sed 's/go//')
|
|
|
|
|
|
if [[ "$CURRENT_VERSION" == "$GO_VERSION" ]]; then
|
|
|
|
|
|
return 0
|
2025-04-28 15:25:10 +02:00
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Old Go Installation ($CURRENT_VERSION) found, upgrading to $GO_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
rm -rf "$GO_INSTALL_DIR"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Go $GO_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local TARBALL="go${GO_VERSION}.linux-${ARCH}.tar.gz"
|
|
|
|
|
|
local URL="https://go.dev/dl/${TARBALL}"
|
|
|
|
|
|
local TMP_TAR=$(mktemp)
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
curl -fsSL "$URL" -o "$TMP_TAR" || {
|
|
|
|
|
|
msg_error "Failed to download $TARBALL"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
tar -C /usr/local -xzf "$TMP_TAR"
|
|
|
|
|
|
ln -sf /usr/local/go/bin/go /usr/local/bin/go
|
|
|
|
|
|
ln -sf /usr/local/go/bin/gofmt /usr/local/bin/gofmt
|
|
|
|
|
|
rm -f "$TMP_TAR"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Go $GO_VERSION"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs Temurin JDK via Adoptium APT repository.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Removes previous JDK if version mismatch
|
|
|
|
|
|
# - Installs or upgrades to specified JAVA_VERSION
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# JAVA_VERSION - Temurin JDK version to install (e.g. 17, 21)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_java() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local JAVA_VERSION="${JAVA_VERSION:-21}"
|
|
|
|
|
|
local DISTRO_CODENAME
|
|
|
|
|
|
DISTRO_CODENAME=$(awk -F= '/VERSION_CODENAME/ { print $2 }' /etc/os-release)
|
|
|
|
|
|
local DESIRED_PACKAGE="temurin-${JAVA_VERSION}-jdk"
|
|
|
|
|
|
|
|
|
|
|
|
# Add Adoptium repo if missing
|
|
|
|
|
|
if [[ ! -f /etc/apt/sources.list.d/adoptium.list ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Setting up Adoptium Repository"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
mkdir -p /etc/apt/keyrings
|
|
|
|
|
|
curl -fsSL "https://packages.adoptium.net/artifactory/api/gpg/key/public" | gpg --dearmor -o /etc/apt/trusted.gpg.d/adoptium.gpg
|
|
|
|
|
|
echo "deb [signed-by=/etc/apt/trusted.gpg.d/adoptium.gpg] https://packages.adoptium.net/artifactory/deb ${DISTRO_CODENAME} main" \
|
|
|
|
|
|
>/etc/apt/sources.list.d/adoptium.list
|
|
|
|
|
|
$STD apt-get update
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_ok "Set up Adoptium Repository"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# Detect currently installed temurin version
|
|
|
|
|
|
local INSTALLED_VERSION=""
|
|
|
|
|
|
if dpkg -l | grep -q "temurin-.*-jdk"; then
|
|
|
|
|
|
INSTALLED_VERSION=$(dpkg -l | awk '/temurin-.*-jdk/{print $2}' | grep -oP 'temurin-\K[0-9]+')
|
|
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if [[ "$INSTALLED_VERSION" == "$JAVA_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Upgrading Temurin JDK $JAVA_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install --only-upgrade -y "$DESIRED_PACKAGE"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_ok "Upgraded Temurin JDK $JAVA_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
2025-04-28 15:25:10 +02:00
|
|
|
|
if [[ -n "$INSTALLED_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Removing Temurin JDK $INSTALLED_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get purge -y "temurin-${INSTALLED_VERSION}-jdk"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Temurin JDK $JAVA_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get install -y "$DESIRED_PACKAGE"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Temurin JDK $JAVA_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
}
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs or updates MongoDB to specified major version.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Preserves data across installations
|
|
|
|
|
|
# - Adds official MongoDB repo
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# MONGO_VERSION - MongoDB major version to install (e.g. 7.0, 8.0)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_mongodb() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local MONGO_VERSION="${MONGO_VERSION:-8.0}"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
local DISTRO_ID DISTRO_CODENAME MONGO_BASE_URL
|
|
|
|
|
|
DISTRO_ID=$(awk -F= '/^ID=/{ gsub(/"/,"",$2); print $2 }' /etc/os-release)
|
|
|
|
|
|
DISTRO_CODENAME=$(awk -F= '/^VERSION_CODENAME=/{ print $2 }' /etc/os-release)
|
|
|
|
|
|
|
2025-07-07 10:59:20 +02:00
|
|
|
|
# Check AVX support
|
|
|
|
|
|
if ! grep -qm1 'avx[^ ]*' /proc/cpuinfo; then
|
|
|
|
|
|
local major="${MONGO_VERSION%%.*}"
|
|
|
|
|
|
if ((major > 5)); then
|
|
|
|
|
|
msg_error "MongoDB ${MONGO_VERSION} requires AVX support, which is not available on this system."
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-05-27 14:29:50 +02:00
|
|
|
|
case "$DISTRO_ID" in
|
2025-06-26 16:27:15 +02:00
|
|
|
|
ubuntu)
|
|
|
|
|
|
MONGO_BASE_URL="https://repo.mongodb.org/apt/ubuntu"
|
|
|
|
|
|
REPO_COMPONENT="multiverse"
|
|
|
|
|
|
;;
|
|
|
|
|
|
debian)
|
|
|
|
|
|
MONGO_BASE_URL="https://repo.mongodb.org/apt/debian"
|
|
|
|
|
|
REPO_COMPONENT="main"
|
|
|
|
|
|
;;
|
2025-05-27 14:29:50 +02:00
|
|
|
|
*)
|
|
|
|
|
|
msg_error "Unsupported distribution: $DISTRO_ID"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
;;
|
|
|
|
|
|
esac
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local REPO_LIST="/etc/apt/sources.list.d/mongodb-org-${MONGO_VERSION}.list"
|
|
|
|
|
|
|
|
|
|
|
|
local INSTALLED_VERSION=""
|
|
|
|
|
|
if command -v mongod >/dev/null; then
|
|
|
|
|
|
INSTALLED_VERSION=$(mongod --version | awk '/db version/{print $3}' | cut -d. -f1,2)
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$INSTALLED_VERSION" == "$MONGO_VERSION" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_info "Upgrading MongoDB $MONGO_VERSION"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
$STD apt-get update
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get install --only-upgrade -y mongodb-org
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD msg_ok "Upgraded MongoDB $MONGO_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
return 0
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -n "$INSTALLED_VERSION" ]]; then
|
|
|
|
|
|
$STD systemctl stop mongod || true
|
|
|
|
|
|
$STD apt-get purge -y mongodb-org || true
|
|
|
|
|
|
rm -f /etc/apt/sources.list.d/mongodb-org-*.list
|
|
|
|
|
|
rm -f /etc/apt/trusted.gpg.d/mongodb-*.gpg
|
|
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup MongoDB $MONGO_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
curl -fsSL "https://pgp.mongodb.com/server-${MONGO_VERSION}.asc" | gpg --dearmor -o "/etc/apt/trusted.gpg.d/mongodb-${MONGO_VERSION}.gpg"
|
2025-06-26 16:27:15 +02:00
|
|
|
|
echo "deb [signed-by=/etc/apt/trusted.gpg.d/mongodb-${MONGO_VERSION}.gpg] ${MONGO_BASE_URL} ${DISTRO_CODENAME}/mongodb-org/${MONGO_VERSION} ${REPO_COMPONENT}" \
|
2025-05-05 15:47:43 +02:00
|
|
|
|
>"$REPO_LIST"
|
|
|
|
|
|
|
2025-05-27 14:29:50 +02:00
|
|
|
|
$STD apt-get update || {
|
|
|
|
|
|
msg_error "APT update failed — invalid MongoDB repo for ${DISTRO_ID}-${DISTRO_CODENAME}?"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
$STD apt-get install -y mongodb-org
|
|
|
|
|
|
|
|
|
|
|
|
mkdir -p /var/lib/mongodb
|
|
|
|
|
|
chown -R mongodb:mongodb /var/lib/mongodb
|
|
|
|
|
|
|
|
|
|
|
|
$STD systemctl enable mongod
|
|
|
|
|
|
$STD systemctl start mongod
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup MongoDB $MONGO_VERSION"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
2025-06-18 12:07:08 +02:00
|
|
|
|
# Downloads and deploys latest GitHub release (source, binary, tarball, asset).
|
2025-05-15 15:18:41 +02:00
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
2025-06-18 12:07:08 +02:00
|
|
|
|
# - Fetches latest release metadata from GitHub API
|
|
|
|
|
|
# - Supports the following modes:
|
|
|
|
|
|
# - tarball: Source code tarball (default if omitted)
|
|
|
|
|
|
# - source: Alias for tarball (same behavior)
|
|
|
|
|
|
# - binary: .deb package install (arch-dependent)
|
|
|
|
|
|
# - prebuild: Prebuilt .tar.gz archive (e.g. Go binaries)
|
|
|
|
|
|
# - singlefile: Standalone binary (no archive, direct chmod +x install)
|
|
|
|
|
|
# - Handles download, extraction/installation and version tracking in ~/.<app>
|
2025-05-15 15:18:41 +02:00
|
|
|
|
#
|
2025-06-18 12:07:08 +02:00
|
|
|
|
# Parameters:
|
|
|
|
|
|
# $1 APP - Application name (used for install path and version file)
|
|
|
|
|
|
# $2 REPO - GitHub repository in form user/repo
|
|
|
|
|
|
# $3 MODE - Release type:
|
|
|
|
|
|
# tarball → source tarball (.tar.gz)
|
|
|
|
|
|
# binary → .deb file (auto-arch matched)
|
|
|
|
|
|
# prebuild → prebuilt archive (e.g. tar.gz)
|
|
|
|
|
|
# singlefile→ standalone binary (chmod +x)
|
|
|
|
|
|
# $4 VERSION - Optional release tag (default: latest)
|
|
|
|
|
|
# $5 TARGET_DIR - Optional install path (default: /opt/<app>)
|
|
|
|
|
|
# $6 ASSET_FILENAME - Required for:
|
|
|
|
|
|
# - prebuild → archive filename or pattern
|
|
|
|
|
|
# - singlefile→ binary filename or pattern
|
|
|
|
|
|
#
|
|
|
|
|
|
# Optional:
|
|
|
|
|
|
# - Set GITHUB_TOKEN env var to increase API rate limit (recommended for CI/CD).
|
|
|
|
|
|
#
|
|
|
|
|
|
# Examples:
|
|
|
|
|
|
# # 1. Minimal: Fetch and deploy source tarball
|
|
|
|
|
|
# fetch_and_deploy_gh_release "myapp" "myuser/myapp"
|
|
|
|
|
|
#
|
|
|
|
|
|
# # 2. Binary install via .deb asset (architecture auto-detected)
|
|
|
|
|
|
# fetch_and_deploy_gh_release "myapp" "myuser/myapp" "binary"
|
|
|
|
|
|
#
|
|
|
|
|
|
# # 3. Prebuilt archive (.tar.gz) with asset filename match
|
|
|
|
|
|
# fetch_and_deploy_gh_release "hanko" "teamhanko/hanko" "prebuild" "latest" "/opt/hanko" "hanko_Linux_x86_64.tar.gz"
|
|
|
|
|
|
#
|
|
|
|
|
|
# # 4. Single binary (chmod +x) like Argus, Promtail etc.
|
|
|
|
|
|
# fetch_and_deploy_gh_release "argus" "release-argus/Argus" "singlefile" "0.26.3" "/opt/argus" "Argus-.*linux-amd64"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function fetch_and_deploy_gh_release() {
|
|
|
|
|
|
local app="$1"
|
|
|
|
|
|
local repo="$2"
|
|
|
|
|
|
local mode="${3:-tarball}" # tarball | binary | prebuild | singlefile
|
|
|
|
|
|
local version="${4:-latest}"
|
|
|
|
|
|
local target="${5:-/opt/$app}"
|
2025-07-06 20:31:17 +02:00
|
|
|
|
local asset_pattern="${6:-}"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
|
|
|
|
|
local app_lc=$(echo "${app,,}" | tr -d ' ')
|
|
|
|
|
|
local version_file="$HOME/.${app_lc}"
|
2025-06-24 13:25:26 +02:00
|
|
|
|
|
2025-07-01 12:31:06 +02:00
|
|
|
|
local api_timeout="--connect-timeout 10 --max-time 60"
|
2025-06-24 13:25:26 +02:00
|
|
|
|
local download_timeout="--connect-timeout 15 --max-time 900"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
|
|
|
|
|
local current_version=""
|
2025-06-24 13:25:26 +02:00
|
|
|
|
[[ -f "$version_file" ]] && current_version=$(<"$version_file")
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if ! command -v jq &>/dev/null; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD apt-get install -y jq &>/dev/null
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
|
|
|
|
|
local api_url="https://api.github.com/repos/$repo/releases"
|
|
|
|
|
|
[[ "$version" != "latest" ]] && api_url="$api_url/tags/$version" || api_url="$api_url/latest"
|
|
|
|
|
|
local header=()
|
2025-05-05 15:47:43 +02:00
|
|
|
|
[[ -n "${GITHUB_TOKEN:-}" ]] && header=(-H "Authorization: token $GITHUB_TOKEN")
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
2025-07-01 12:31:06 +02:00
|
|
|
|
# dns pre check
|
|
|
|
|
|
local gh_host
|
|
|
|
|
|
gh_host=$(awk -F/ '{print $3}' <<<"$api_url")
|
|
|
|
|
|
if ! getent hosts "$gh_host" &>/dev/null; then
|
|
|
|
|
|
msg_error "DNS resolution failed for $gh_host – check /etc/resolv.conf or networking"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-24 13:25:26 +02:00
|
|
|
|
local max_retries=3 retry_delay=2 attempt=1 success=false resp http_code
|
|
|
|
|
|
|
|
|
|
|
|
while ((attempt <= max_retries)); do
|
2025-07-22 14:18:34 +02:00
|
|
|
|
resp=$(curl $api_timeout -fsSL -w "%{http_code}" -o /tmp/gh_rel.json "${header[@]}" "$api_url") && success=true && break
|
2025-06-24 13:25:26 +02:00
|
|
|
|
sleep "$retry_delay"
|
|
|
|
|
|
((attempt++))
|
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
|
|
if ! $success; then
|
2025-07-01 12:31:06 +02:00
|
|
|
|
msg_error "Failed to fetch release metadata from $api_url after $max_retries attempts"
|
2025-06-24 13:25:26 +02:00
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
http_code="${resp:(-3)}"
|
|
|
|
|
|
[[ "$http_code" != "200" ]] && {
|
2025-06-24 13:25:26 +02:00
|
|
|
|
msg_error "GitHub API returned HTTP $http_code"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
local json tag_name
|
|
|
|
|
|
json=$(</tmp/gh_rel.json)
|
|
|
|
|
|
tag_name=$(echo "$json" | jq -r '.tag_name // .name // empty')
|
|
|
|
|
|
[[ "$tag_name" =~ ^v ]] && version="${tag_name:1}" || version="$tag_name"
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$current_version" == "$version" ]]; then
|
|
|
|
|
|
$STD msg_ok "$app is already up-to-date (v$version)"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
return 0
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
local tmpdir
|
|
|
|
|
|
tmpdir=$(mktemp -d) || return 1
|
2025-06-18 12:07:08 +02:00
|
|
|
|
local filename="" url=""
|
|
|
|
|
|
|
2025-06-24 13:25:26 +02:00
|
|
|
|
msg_info "Fetching GitHub release: $app ($version)"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
2025-07-03 15:16:42 +02:00
|
|
|
|
### Tarball Mode ###
|
2025-06-18 12:07:08 +02:00
|
|
|
|
if [[ "$mode" == "tarball" || "$mode" == "source" ]]; then
|
|
|
|
|
|
url=$(echo "$json" | jq -r '.tarball_url // empty')
|
|
|
|
|
|
[[ -z "$url" ]] && url="https://github.com/$repo/archive/refs/tags/v$version.tar.gz"
|
|
|
|
|
|
filename="${app_lc}-${version}.tar.gz"
|
|
|
|
|
|
|
2025-07-22 14:18:34 +02:00
|
|
|
|
curl $download_timeout -fsSL -o "$tmpdir/$filename" "$url" || {
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_error "Download failed: $url"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
mkdir -p "$target"
|
|
|
|
|
|
tar -xzf "$tmpdir/$filename" -C "$tmpdir"
|
|
|
|
|
|
local unpack_dir
|
|
|
|
|
|
unpack_dir=$(find "$tmpdir" -mindepth 1 -maxdepth 1 -type d | head -n1)
|
|
|
|
|
|
|
|
|
|
|
|
shopt -s dotglob nullglob
|
|
|
|
|
|
cp -r "$unpack_dir"/* "$target/"
|
|
|
|
|
|
shopt -u dotglob nullglob
|
|
|
|
|
|
|
2025-07-03 15:16:42 +02:00
|
|
|
|
### Binary Mode ###
|
2025-06-18 12:07:08 +02:00
|
|
|
|
elif [[ "$mode" == "binary" ]]; then
|
|
|
|
|
|
local arch
|
|
|
|
|
|
arch=$(dpkg --print-architecture 2>/dev/null || uname -m)
|
2025-06-30 20:23:25 +02:00
|
|
|
|
[[ "$arch" == "x86_64" ]] && arch="amd64"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
[[ "$arch" == "aarch64" ]] && arch="arm64"
|
|
|
|
|
|
|
|
|
|
|
|
local assets url_match=""
|
|
|
|
|
|
assets=$(echo "$json" | jq -r '.assets[].browser_download_url')
|
|
|
|
|
|
|
2025-07-03 15:16:42 +02:00
|
|
|
|
# If explicit filename pattern is provided (param $6), match that first
|
2025-07-06 20:39:15 +02:00
|
|
|
|
if [[ -n "$asset_pattern" ]]; then
|
2025-06-30 20:23:25 +02:00
|
|
|
|
for u in $assets; do
|
2025-07-06 20:39:15 +02:00
|
|
|
|
[[ "$u" =~ $asset_pattern || "$u" == *"$asset_pattern" ]] && url_match="$u" && break
|
2025-06-30 20:23:25 +02:00
|
|
|
|
done
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-07-03 15:16:42 +02:00
|
|
|
|
# If no match via explicit pattern, fall back to architecture heuristic
|
2025-06-30 20:23:25 +02:00
|
|
|
|
if [[ -z "$url_match" ]]; then
|
|
|
|
|
|
for u in $assets; do
|
|
|
|
|
|
if [[ "$u" =~ ($arch|amd64|x86_64|aarch64|arm64).*\.deb$ ]]; then
|
|
|
|
|
|
url_match="$u"
|
|
|
|
|
|
break
|
|
|
|
|
|
fi
|
|
|
|
|
|
done
|
|
|
|
|
|
fi
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
2025-07-03 15:16:42 +02:00
|
|
|
|
# Fallback: any .deb file
|
2025-06-18 12:07:08 +02:00
|
|
|
|
if [[ -z "$url_match" ]]; then
|
|
|
|
|
|
for u in $assets; do
|
|
|
|
|
|
[[ "$u" =~ \.deb$ ]] && url_match="$u" && break
|
|
|
|
|
|
done
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -z "$url_match" ]]; then
|
|
|
|
|
|
msg_error "No suitable .deb asset found for $app"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
filename="${url_match##*/}"
|
2025-07-22 14:18:34 +02:00
|
|
|
|
curl $download_timeout -fsSL -o "$tmpdir/$filename" "$url_match" || {
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_error "Download failed: $url_match"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
chmod 644 "$tmpdir/$filename"
|
|
|
|
|
|
$STD apt-get install -y "$tmpdir/$filename" || {
|
|
|
|
|
|
$STD dpkg -i "$tmpdir/$filename" || {
|
|
|
|
|
|
msg_error "Both apt and dpkg installation failed"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-09 14:06:33 +02:00
|
|
|
|
### Prebuild Mode ###
|
2025-06-18 12:07:08 +02:00
|
|
|
|
elif [[ "$mode" == "prebuild" ]]; then
|
2025-07-03 15:16:42 +02:00
|
|
|
|
local pattern="${6%\"}"
|
|
|
|
|
|
pattern="${pattern#\"}"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
[[ -z "$pattern" ]] && {
|
|
|
|
|
|
msg_error "Mode 'prebuild' requires 6th parameter (asset filename pattern)"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
local asset_url=""
|
|
|
|
|
|
for u in $(echo "$json" | jq -r '.assets[].browser_download_url'); do
|
2025-07-03 15:16:42 +02:00
|
|
|
|
filename_candidate="${u##*/}"
|
|
|
|
|
|
case "$filename_candidate" in
|
|
|
|
|
|
$pattern)
|
|
|
|
|
|
asset_url="$u"
|
|
|
|
|
|
break
|
|
|
|
|
|
;;
|
|
|
|
|
|
esac
|
2025-05-05 15:47:43 +02:00
|
|
|
|
done
|
2025-05-14 16:08:21 +02:00
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
[[ -z "$asset_url" ]] && {
|
|
|
|
|
|
msg_error "No asset matching '$pattern' found"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
filename="${asset_url##*/}"
|
2025-07-22 14:18:34 +02:00
|
|
|
|
curl $download_timeout -fsSL -o "$tmpdir/$filename" "$asset_url" || {
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_error "Download failed: $asset_url"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-09 14:06:33 +02:00
|
|
|
|
local unpack_tmp
|
|
|
|
|
|
unpack_tmp=$(mktemp -d)
|
2025-06-18 12:07:08 +02:00
|
|
|
|
mkdir -p "$target"
|
2025-07-09 14:06:33 +02:00
|
|
|
|
|
2025-07-08 17:19:51 +02:00
|
|
|
|
if [[ "$filename" == *.zip ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
if ! command -v unzip &>/dev/null; then
|
|
|
|
|
|
$STD apt-get install -y unzip
|
|
|
|
|
|
fi
|
2025-07-09 14:06:33 +02:00
|
|
|
|
unzip -q "$tmpdir/$filename" -d "$unpack_tmp"
|
2025-07-03 15:16:42 +02:00
|
|
|
|
elif [[ "$filename" == *.tar.* ]]; then
|
2025-07-09 14:06:33 +02:00
|
|
|
|
tar -xf "$tmpdir/$filename" -C "$unpack_tmp"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
else
|
|
|
|
|
|
msg_error "Unsupported archive format: $filename"
|
2025-07-09 14:06:33 +02:00
|
|
|
|
rm -rf "$tmpdir" "$unpack_tmp"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
return 1
|
2025-05-14 16:08:21 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
2025-07-09 14:06:33 +02:00
|
|
|
|
local top_dirs
|
|
|
|
|
|
top_dirs=$(find "$unpack_tmp" -mindepth 1 -maxdepth 1 -type d | wc -l)
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$top_dirs" -eq 1 ]]; then
|
|
|
|
|
|
# Strip leading folder
|
|
|
|
|
|
local inner_dir
|
|
|
|
|
|
inner_dir=$(find "$unpack_tmp" -mindepth 1 -maxdepth 1 -type d)
|
|
|
|
|
|
shopt -s dotglob nullglob
|
|
|
|
|
|
cp -r "$inner_dir"/* "$target/"
|
|
|
|
|
|
shopt -u dotglob nullglob
|
|
|
|
|
|
else
|
|
|
|
|
|
# Copy all contents
|
|
|
|
|
|
shopt -s dotglob nullglob
|
|
|
|
|
|
cp -r "$unpack_tmp"/* "$target/"
|
|
|
|
|
|
shopt -u dotglob nullglob
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
rm -rf "$unpack_tmp"
|
|
|
|
|
|
|
2025-07-03 15:16:42 +02:00
|
|
|
|
### Singlefile Mode ###
|
2025-06-18 12:07:08 +02:00
|
|
|
|
elif [[ "$mode" == "singlefile" ]]; then
|
2025-07-03 15:16:42 +02:00
|
|
|
|
local pattern="${6%\"}"
|
|
|
|
|
|
pattern="${pattern#\"}"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
[[ -z "$pattern" ]] && {
|
|
|
|
|
|
msg_error "Mode 'singlefile' requires 6th parameter (asset filename pattern)"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
local asset_url=""
|
|
|
|
|
|
for u in $(echo "$json" | jq -r '.assets[].browser_download_url'); do
|
2025-07-03 15:16:42 +02:00
|
|
|
|
filename_candidate="${u##*/}"
|
|
|
|
|
|
case "$filename_candidate" in
|
|
|
|
|
|
$pattern)
|
|
|
|
|
|
asset_url="$u"
|
|
|
|
|
|
break
|
|
|
|
|
|
;;
|
|
|
|
|
|
esac
|
2025-06-18 12:07:08 +02:00
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
|
|
[[ -z "$asset_url" ]] && {
|
|
|
|
|
|
msg_error "No asset matching '$pattern' found"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
filename="${asset_url##*/}"
|
|
|
|
|
|
mkdir -p "$target"
|
2025-07-08 17:19:51 +02:00
|
|
|
|
|
|
|
|
|
|
local use_filename="${USE_ORIGINAL_FILENAME:-false}"
|
|
|
|
|
|
local target_file="$app"
|
|
|
|
|
|
[[ "$use_filename" == "true" ]] && target_file="$filename"
|
|
|
|
|
|
|
2025-07-22 14:18:34 +02:00
|
|
|
|
curl $download_timeout -fsSL -o "$target/$target_file" "$asset_url" || {
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_error "Download failed: $asset_url"
|
|
|
|
|
|
rm -rf "$tmpdir"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-22 14:18:34 +02:00
|
|
|
|
if [[ "$tagrget_file" != *.jar && -f "$target/$target_file" ]]; then
|
2025-07-08 17:19:51 +02:00
|
|
|
|
chmod +x "$target/$target_file"
|
|
|
|
|
|
fi
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
msg_error "Unknown mode: $mode"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
rm -rf "$tmpdir"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
|
|
|
|
|
echo "$version" >"$version_file"
|
2025-06-24 13:25:26 +02:00
|
|
|
|
msg_ok "Deployed: $app ($version)"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
rm -rf "$tmpdir"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs a local IP updater script using networkd-dispatcher.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Stores current IP in /run/local-ip.env
|
|
|
|
|
|
# - Automatically runs on network changes
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_local_ip_helper() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local BASE_DIR="/usr/local/community-scripts/ip-management"
|
|
|
|
|
|
local SCRIPT_PATH="$BASE_DIR/update_local_ip.sh"
|
|
|
|
|
|
local IP_FILE="/run/local-ip.env"
|
|
|
|
|
|
local DISPATCHER_SCRIPT="/etc/networkd-dispatcher/routable.d/10-update-local-ip.sh"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
mkdir -p "$BASE_DIR"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# Install networkd-dispatcher if not present
|
|
|
|
|
|
if ! dpkg -s networkd-dispatcher >/dev/null 2>&1; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y networkd-dispatcher
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# Write update_local_ip.sh
|
|
|
|
|
|
cat <<'EOF' >"$SCRIPT_PATH"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
#!/bin/bash
|
|
|
|
|
|
set -euo pipefail
|
|
|
|
|
|
|
|
|
|
|
|
IP_FILE="/run/local-ip.env"
|
|
|
|
|
|
mkdir -p "$(dirname "$IP_FILE")"
|
|
|
|
|
|
|
|
|
|
|
|
get_current_ip() {
|
|
|
|
|
|
local targets=("8.8.8.8" "1.1.1.1" "192.168.1.1" "10.0.0.1" "172.16.0.1" "default")
|
|
|
|
|
|
local ip
|
|
|
|
|
|
|
|
|
|
|
|
for target in "${targets[@]}"; do
|
|
|
|
|
|
if [[ "$target" == "default" ]]; then
|
|
|
|
|
|
ip=$(ip route get 1 2>/dev/null | awk '{for(i=1;i<=NF;i++) if ($i=="src") print $(i+1)}')
|
|
|
|
|
|
else
|
|
|
|
|
|
ip=$(ip route get "$target" 2>/dev/null | awk '{for(i=1;i<=NF;i++) if ($i=="src") print $(i+1)}')
|
|
|
|
|
|
fi
|
|
|
|
|
|
if [[ -n "$ip" ]]; then
|
|
|
|
|
|
echo "$ip"
|
|
|
|
|
|
return 0
|
|
|
|
|
|
fi
|
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
current_ip="$(get_current_ip)"
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -z "$current_ip" ]]; then
|
|
|
|
|
|
echo "[ERROR] Could not detect local IP" >&2
|
|
|
|
|
|
exit 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -f "$IP_FILE" ]]; then
|
|
|
|
|
|
source "$IP_FILE"
|
|
|
|
|
|
[[ "$LOCAL_IP" == "$current_ip" ]] && exit 0
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
echo "LOCAL_IP=$current_ip" > "$IP_FILE"
|
|
|
|
|
|
echo "[INFO] LOCAL_IP updated to $current_ip"
|
|
|
|
|
|
EOF
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
chmod +x "$SCRIPT_PATH"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# Install dispatcher hook
|
|
|
|
|
|
mkdir -p "$(dirname "$DISPATCHER_SCRIPT")"
|
|
|
|
|
|
cat <<EOF >"$DISPATCHER_SCRIPT"
|
2025-04-28 15:25:10 +02:00
|
|
|
|
#!/bin/bash
|
|
|
|
|
|
$SCRIPT_PATH
|
|
|
|
|
|
EOF
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
chmod +x "$DISPATCHER_SCRIPT"
|
2025-05-14 10:00:41 +02:00
|
|
|
|
systemctl enable -q --now networkd-dispatcher.service
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Loads LOCAL_IP from persistent store or detects if missing.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Loads from /run/local-ip.env or performs runtime lookup
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function import_local_ip() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local IP_FILE="/run/local-ip.env"
|
|
|
|
|
|
if [[ -f "$IP_FILE" ]]; then
|
|
|
|
|
|
# shellcheck disable=SC1090
|
|
|
|
|
|
source "$IP_FILE"
|
|
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if [[ -z "${LOCAL_IP:-}" ]]; then
|
|
|
|
|
|
get_current_ip() {
|
|
|
|
|
|
local targets=("8.8.8.8" "1.1.1.1" "192.168.1.1" "10.0.0.1" "172.16.0.1" "default")
|
|
|
|
|
|
local ip
|
|
|
|
|
|
|
|
|
|
|
|
for target in "${targets[@]}"; do
|
|
|
|
|
|
if [[ "$target" == "default" ]]; then
|
|
|
|
|
|
ip=$(ip route get 1 2>/dev/null | awk '{for(i=1;i<=NF;i++) if ($i=="src") print $(i+1)}')
|
|
|
|
|
|
else
|
|
|
|
|
|
ip=$(ip route get "$target" 2>/dev/null | awk '{for(i=1;i<=NF;i++) if ($i=="src") print $(i+1)}')
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if [[ -n "$ip" ]]; then
|
|
|
|
|
|
echo "$ip"
|
|
|
|
|
|
return 0
|
|
|
|
|
|
fi
|
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LOCAL_IP="$(get_current_ip || true)"
|
|
|
|
|
|
if [[ -z "$LOCAL_IP" ]]; then
|
|
|
|
|
|
msg_error "Could not determine LOCAL_IP"
|
|
|
|
|
|
return 1
|
2025-04-28 15:25:10 +02:00
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-04-28 15:25:10 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
export LOCAL_IP
|
2025-04-28 15:25:10 +02:00
|
|
|
|
}
|
2025-04-29 16:04:29 +02:00
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Downloads file with optional progress indicator using pv.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Arguments:
|
|
|
|
|
|
# $1 - URL
|
|
|
|
|
|
# $2 - Destination path
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
function download_with_progress() {
|
|
|
|
|
|
local url="$1"
|
|
|
|
|
|
local output="$2"
|
|
|
|
|
|
if [ -n "$SPINNER_PID" ] && ps -p "$SPINNER_PID" >/dev/null; then kill "$SPINNER_PID" >/dev/null; fi
|
|
|
|
|
|
|
|
|
|
|
|
if ! command -v pv &>/dev/null; then
|
|
|
|
|
|
$STD apt-get install -y pv
|
|
|
|
|
|
fi
|
|
|
|
|
|
set -o pipefail
|
2025-04-29 16:04:29 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
# Content-Length aus HTTP-Header holen
|
|
|
|
|
|
local content_length
|
|
|
|
|
|
content_length=$(curl -fsSLI "$url" | awk '/Content-Length/ {print $2}' | tr -d '\r' || true)
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -z "$content_length" ]]; then
|
|
|
|
|
|
if ! curl -fL# -o "$output" "$url"; then
|
|
|
|
|
|
msg_error "Download failed"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
else
|
|
|
|
|
|
if ! curl -fsSL "$url" | pv -s "$content_length" >"$output"; then
|
|
|
|
|
|
msg_error "Download failed"
|
|
|
|
|
|
return 1
|
2025-04-29 16:04:29 +02:00
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
|
|
|
|
|
}
|
2025-04-29 16:04:29 +02:00
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs or upgrades uv (Python package manager) from GitHub releases.
|
2025-06-18 12:07:08 +02:00
|
|
|
|
# - Downloads platform-specific tarball (no install.sh!)
|
|
|
|
|
|
# - Extracts uv binary
|
|
|
|
|
|
# - Places it in /usr/local/bin
|
|
|
|
|
|
# - Optionally installs a specific Python version via uv
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
function setup_uv() {
|
2025-06-18 12:07:08 +02:00
|
|
|
|
local UV_BIN="/usr/local/bin/uv"
|
|
|
|
|
|
local TMP_DIR
|
2025-05-05 15:47:43 +02:00
|
|
|
|
TMP_DIR=$(mktemp -d)
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
|
|
|
|
|
# Determine system architecture
|
|
|
|
|
|
local ARCH
|
2025-05-05 15:47:43 +02:00
|
|
|
|
ARCH=$(uname -m)
|
2025-06-18 12:07:08 +02:00
|
|
|
|
local UV_TAR
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
case "$ARCH" in
|
|
|
|
|
|
x86_64) UV_TAR="uv-x86_64-unknown-linux-gnu.tar.gz" ;;
|
|
|
|
|
|
aarch64) UV_TAR="uv-aarch64-unknown-linux-gnu.tar.gz" ;;
|
|
|
|
|
|
*)
|
2025-05-05 15:47:43 +02:00
|
|
|
|
msg_error "Unsupported architecture: $ARCH"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
2025-06-18 12:07:08 +02:00
|
|
|
|
;;
|
|
|
|
|
|
esac
|
|
|
|
|
|
|
|
|
|
|
|
# Get latest version from GitHub
|
|
|
|
|
|
local LATEST_VERSION
|
|
|
|
|
|
LATEST_VERSION=$(curl -fsSL https://api.github.com/repos/astral-sh/uv/releases/latest |
|
|
|
|
|
|
grep '"tag_name":' | cut -d '"' -f4 | sed 's/^v//')
|
2025-05-05 15:47:43 +02:00
|
|
|
|
|
|
|
|
|
|
if [[ -z "$LATEST_VERSION" ]]; then
|
|
|
|
|
|
msg_error "Could not fetch latest uv version from GitHub."
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
# Check if uv is already up to date
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if [[ -x "$UV_BIN" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
local INSTALLED_VERSION
|
2025-05-05 15:47:43 +02:00
|
|
|
|
INSTALLED_VERSION=$($UV_BIN -V | awk '{print $2}')
|
|
|
|
|
|
if [[ "$INSTALLED_VERSION" == "$LATEST_VERSION" ]]; then
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
[[ ":$PATH:" != *":/usr/local/bin:"* ]] && export PATH="/usr/local/bin:$PATH"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
return 0
|
2025-04-29 16:04:29 +02:00
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Updating uv from $INSTALLED_VERSION to $LATEST_VERSION"
|
2025-04-29 16:04:29 +02:00
|
|
|
|
fi
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup uv $LATEST_VERSION"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
fi
|
2025-04-29 16:04:29 +02:00
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
# Download and install manually
|
|
|
|
|
|
local UV_URL="https://github.com/astral-sh/uv/releases/latest/download/${UV_TAR}"
|
|
|
|
|
|
if ! curl -fsSL "$UV_URL" -o "$TMP_DIR/uv.tar.gz"; then
|
|
|
|
|
|
msg_error "Failed to download $UV_URL"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if ! tar -xzf "$TMP_DIR/uv.tar.gz" -C "$TMP_DIR"; then
|
|
|
|
|
|
msg_error "Failed to extract uv archive"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
install -m 755 "$TMP_DIR"/*/uv "$UV_BIN" || {
|
|
|
|
|
|
msg_error "Failed to install uv binary"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
2025-04-29 16:04:29 +02:00
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
rm -rf "$TMP_DIR"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
ensure_usr_local_bin_persist
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup uv $LATEST_VERSION"
|
|
|
|
|
|
|
|
|
|
|
|
# Optional: install specific Python version
|
|
|
|
|
|
if [[ -n "${PYTHON_VERSION:-}" ]]; then
|
|
|
|
|
|
local VERSION_MATCH
|
|
|
|
|
|
VERSION_MATCH=$(uv python list --only-downloads |
|
|
|
|
|
|
grep -E "^cpython-${PYTHON_VERSION//./\\.}\.[0-9]+-linux" |
|
|
|
|
|
|
cut -d'-' -f2 | sort -V | tail -n1)
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -z "$VERSION_MATCH" ]]; then
|
|
|
|
|
|
msg_error "No matching Python $PYTHON_VERSION.x version found via uv"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if ! uv python list | grep -q "cpython-${VERSION_MATCH}-linux.*uv/python"; then
|
|
|
|
|
|
if ! $STD uv python install "$VERSION_MATCH"; then
|
|
|
|
|
|
msg_error "Failed to install Python $VERSION_MATCH via uv"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
msg_ok "Setup Python $VERSION_MATCH via uv"
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
2025-04-29 16:04:29 +02:00
|
|
|
|
}
|
2025-04-29 16:20:23 +02:00
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Ensures /usr/local/bin is permanently in system PATH.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Adds to /etc/profile.d if not present
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-04-29 16:20:23 +02:00
|
|
|
|
function ensure_usr_local_bin_persist() {
|
2025-05-05 15:47:43 +02:00
|
|
|
|
local PROFILE_FILE="/etc/profile.d/custom_path.sh"
|
2025-04-29 16:20:23 +02:00
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
if [[ ! -f "$PROFILE_FILE" ]] && ! command -v pveversion &>/dev/null; then
|
|
|
|
|
|
echo 'export PATH="/usr/local/bin:$PATH"' >"$PROFILE_FILE"
|
|
|
|
|
|
chmod +x "$PROFILE_FILE"
|
|
|
|
|
|
fi
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-15 15:18:41 +02:00
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs or updates Ghostscript (gs) from source.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Fetches latest release
|
|
|
|
|
|
# - Builds and installs system-wide
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-05-05 15:47:43 +02:00
|
|
|
|
function setup_gs() {
|
|
|
|
|
|
mkdir -p /tmp
|
|
|
|
|
|
TMP_DIR=$(mktemp -d)
|
|
|
|
|
|
CURRENT_VERSION=$(gs --version 2>/dev/null || echo "0")
|
|
|
|
|
|
|
|
|
|
|
|
RELEASE_JSON=$(curl -fsSL https://api.github.com/repos/ArtifexSoftware/ghostpdl-downloads/releases/latest)
|
|
|
|
|
|
LATEST_VERSION=$(echo "$RELEASE_JSON" | grep '"tag_name":' | head -n1 | cut -d '"' -f4 | sed 's/^gs//')
|
|
|
|
|
|
LATEST_VERSION_DOTTED=$(echo "$RELEASE_JSON" | grep '"name":' | head -n1 | grep -o '[0-9]\+\.[0-9]\+\.[0-9]\+')
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -z "$LATEST_VERSION" ]]; then
|
|
|
|
|
|
msg_error "Could not determine latest Ghostscript version from GitHub."
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if dpkg --compare-versions "$CURRENT_VERSION" ge "$LATEST_VERSION_DOTTED"; then
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Ghostscript $LATEST_VERSION_DOTTED"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
curl -fsSL "https://github.com/ArtifexSoftware/ghostpdl-downloads/releases/download/gs${LATEST_VERSION}/ghostscript-${LATEST_VERSION_DOTTED}.tar.gz" -o "$TMP_DIR/ghostscript.tar.gz"
|
|
|
|
|
|
|
|
|
|
|
|
if ! tar -xzf "$TMP_DIR/ghostscript.tar.gz" -C "$TMP_DIR"; then
|
|
|
|
|
|
msg_error "Failed to extract Ghostscript archive."
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
cd "$TMP_DIR/ghostscript-${LATEST_VERSION_DOTTED}" || {
|
|
|
|
|
|
msg_error "Failed to enter Ghostscript source directory."
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
}
|
|
|
|
|
|
$STD apt-get install -y build-essential libpng-dev zlib1g-dev
|
|
|
|
|
|
./configure >/dev/null && make && sudo make install >/dev/null
|
|
|
|
|
|
local EXIT_CODE=$?
|
|
|
|
|
|
hash -r
|
|
|
|
|
|
if [[ ! -x "$(command -v gs)" ]]; then
|
|
|
|
|
|
if [[ -x /usr/local/bin/gs ]]; then
|
|
|
|
|
|
ln -sf /usr/local/bin/gs /usr/bin/gs
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
|
|
|
|
|
|
if [[ $EXIT_CODE -eq 0 ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Ghostscript $LATEST_VERSION_DOTTED"
|
2025-05-05 15:47:43 +02:00
|
|
|
|
else
|
|
|
|
|
|
msg_error "Ghostscript installation failed"
|
|
|
|
|
|
fi
|
2025-04-29 16:20:23 +02:00
|
|
|
|
}
|
2025-05-15 15:18:41 +02:00
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs rbenv and ruby-build, installs Ruby and optionally Rails.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Downloads rbenv and ruby-build from GitHub
|
|
|
|
|
|
# - Compiles and installs target Ruby version
|
|
|
|
|
|
# - Optionally installs Rails via gem
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# RUBY_VERSION - Ruby version to install (default: 3.4.4)
|
|
|
|
|
|
# RUBY_INSTALL_RAILS - true/false to install Rails (default: true)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_ruby() {
|
2025-05-15 15:18:41 +02:00
|
|
|
|
local RUBY_VERSION="${RUBY_VERSION:-3.4.4}"
|
|
|
|
|
|
local RUBY_INSTALL_RAILS="${RUBY_INSTALL_RAILS:-true}"
|
|
|
|
|
|
|
|
|
|
|
|
local RBENV_DIR="$HOME/.rbenv"
|
|
|
|
|
|
local RBENV_BIN="$RBENV_DIR/bin/rbenv"
|
|
|
|
|
|
local PROFILE_FILE="$HOME/.profile"
|
|
|
|
|
|
local TMP_DIR
|
|
|
|
|
|
TMP_DIR=$(mktemp -d)
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Ruby $RUBY_VERSION"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
|
|
|
|
|
|
local RBENV_RELEASE
|
|
|
|
|
|
RBENV_RELEASE=$(curl -fsSL https://api.github.com/repos/rbenv/rbenv/releases/latest | grep '"tag_name":' | cut -d '"' -f4 | sed 's/^v//')
|
|
|
|
|
|
if [[ -z "$RBENV_RELEASE" ]]; then
|
|
|
|
|
|
msg_error "Failed to fetch latest rbenv version"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
curl -fsSL "https://github.com/rbenv/rbenv/archive/refs/tags/v${RBENV_RELEASE}.tar.gz" -o "$TMP_DIR/rbenv.tar.gz"
|
|
|
|
|
|
tar -xzf "$TMP_DIR/rbenv.tar.gz" -C "$TMP_DIR"
|
|
|
|
|
|
mkdir -p "$RBENV_DIR"
|
|
|
|
|
|
cp -r "$TMP_DIR/rbenv-${RBENV_RELEASE}/." "$RBENV_DIR/"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
cd "$RBENV_DIR" && src/configure && $STD make -C src
|
2025-05-15 15:18:41 +02:00
|
|
|
|
|
|
|
|
|
|
local RUBY_BUILD_RELEASE
|
|
|
|
|
|
RUBY_BUILD_RELEASE=$(curl -fsSL https://api.github.com/repos/rbenv/ruby-build/releases/latest | grep '"tag_name":' | cut -d '"' -f4 | sed 's/^v//')
|
|
|
|
|
|
if [[ -z "$RUBY_BUILD_RELEASE" ]]; then
|
|
|
|
|
|
msg_error "Failed to fetch latest ruby-build version"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
curl -fsSL "https://github.com/rbenv/ruby-build/archive/refs/tags/v${RUBY_BUILD_RELEASE}.tar.gz" -o "$TMP_DIR/ruby-build.tar.gz"
|
|
|
|
|
|
tar -xzf "$TMP_DIR/ruby-build.tar.gz" -C "$TMP_DIR"
|
|
|
|
|
|
mkdir -p "$RBENV_DIR/plugins/ruby-build"
|
|
|
|
|
|
cp -r "$TMP_DIR/ruby-build-${RUBY_BUILD_RELEASE}/." "$RBENV_DIR/plugins/ruby-build/"
|
|
|
|
|
|
echo "$RUBY_BUILD_RELEASE" >"$RBENV_DIR/plugins/ruby-build/RUBY_BUILD_version.txt"
|
|
|
|
|
|
|
|
|
|
|
|
if ! grep -q 'rbenv init' "$PROFILE_FILE"; then
|
|
|
|
|
|
echo 'export PATH="$HOME/.rbenv/bin:$PATH"' >>"$PROFILE_FILE"
|
|
|
|
|
|
echo 'eval "$(rbenv init -)"' >>"$PROFILE_FILE"
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
export PATH="$RBENV_DIR/bin:$PATH"
|
|
|
|
|
|
eval "$("$RBENV_BIN" init - bash)"
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
if ! "$RBENV_BIN" versions --bare | grep -qx "$RUBY_VERSION"; then
|
2025-05-15 15:18:41 +02:00
|
|
|
|
$STD "$RBENV_BIN" install "$RUBY_VERSION"
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
"$RBENV_BIN" global "$RUBY_VERSION"
|
|
|
|
|
|
hash -r
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$RUBY_INSTALL_RAILS" == "true" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Rails via gem"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
gem install rails
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Rails $(rails -v)"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Ruby $RUBY_VERSION"
|
2025-05-15 15:18:41 +02:00
|
|
|
|
}
|
2025-05-18 09:27:55 +02:00
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Creates and installs self-signed certificates.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Create a self-signed certificate with option to override application name
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# APP - Application name (default: $APPLICATION variable)
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function create_selfsigned_certs() {
|
2025-05-18 09:27:55 +02:00
|
|
|
|
local app=${APP:-$(echo "${APPLICATION,,}" | tr -d ' ')}
|
|
|
|
|
|
$STD openssl req -x509 -nodes -days 365 -newkey rsa:4096 \
|
|
|
|
|
|
-keyout /etc/ssl/private/"$app"-selfsigned.key \
|
|
|
|
|
|
-out /etc/ssl/certs/"$app"-selfsigned.crt \
|
|
|
|
|
|
-subj "/C=US/O=$app/OU=Domain Control Validated/CN=localhost"
|
|
|
|
|
|
}
|
2025-05-27 14:29:50 +02:00
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs Rust toolchain and optional global crates via cargo.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Installs rustup (if missing)
|
|
|
|
|
|
# - Installs or updates desired Rust toolchain (stable, nightly, or versioned)
|
|
|
|
|
|
# - Installs or updates specified global crates using `cargo install`
|
|
|
|
|
|
#
|
|
|
|
|
|
# Notes:
|
|
|
|
|
|
# - Skips crate install if exact version is already present
|
|
|
|
|
|
# - Updates crate if newer version or different version is requested
|
|
|
|
|
|
#
|
|
|
|
|
|
# Variables:
|
|
|
|
|
|
# RUST_TOOLCHAIN - Rust toolchain to install (default: stable)
|
|
|
|
|
|
# RUST_CRATES - Comma-separated list of crates (e.g. "cargo-edit,wasm-pack@0.12.1")
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_rust() {
|
2025-05-27 14:29:50 +02:00
|
|
|
|
local RUST_TOOLCHAIN="${RUST_TOOLCHAIN:-stable}"
|
|
|
|
|
|
local RUST_CRATES="${RUST_CRATES:-}"
|
|
|
|
|
|
local CARGO_BIN="${HOME}/.cargo/bin"
|
|
|
|
|
|
|
|
|
|
|
|
# rustup & toolchain
|
|
|
|
|
|
if ! command -v rustup &>/dev/null; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Rust"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
curl -fsSL https://sh.rustup.rs | $STD sh -s -- -y --default-toolchain "$RUST_TOOLCHAIN"
|
|
|
|
|
|
export PATH="$CARGO_BIN:$PATH"
|
|
|
|
|
|
echo 'export PATH="$HOME/.cargo/bin:$PATH"' >>"$HOME/.profile"
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Rust"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
else
|
|
|
|
|
|
$STD rustup install "$RUST_TOOLCHAIN"
|
|
|
|
|
|
$STD rustup default "$RUST_TOOLCHAIN"
|
|
|
|
|
|
$STD rustup update "$RUST_TOOLCHAIN"
|
|
|
|
|
|
msg_ok "Rust toolchain set to $RUST_TOOLCHAIN"
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
# install/update crates
|
|
|
|
|
|
if [[ -n "$RUST_CRATES" ]]; then
|
|
|
|
|
|
IFS=',' read -ra CRATES <<<"$RUST_CRATES"
|
|
|
|
|
|
for crate in "${CRATES[@]}"; do
|
|
|
|
|
|
local NAME VER INSTALLED_VER
|
|
|
|
|
|
if [[ "$crate" == *"@"* ]]; then
|
|
|
|
|
|
NAME="${crate%@*}"
|
|
|
|
|
|
VER="${crate##*@}"
|
|
|
|
|
|
else
|
|
|
|
|
|
NAME="$crate"
|
|
|
|
|
|
VER=""
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
INSTALLED_VER=$(cargo install --list 2>/dev/null | awk "/^$NAME v[0-9]/ {print \$2}" | tr -d 'v')
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -n "$INSTALLED_VER" ]]; then
|
|
|
|
|
|
if [[ -n "$VER" && "$VER" != "$INSTALLED_VER" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Update $NAME: $INSTALLED_VER → $VER"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
$STD cargo install "$NAME" --version "$VER" --force
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Updated $NAME to $VER"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
elif [[ -z "$VER" ]]; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Update $NAME: $INSTALLED_VER → latest"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
$STD cargo install "$NAME" --force
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Updated $NAME to latest"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
fi
|
|
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup $NAME ${VER:+($VER)}"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
$STD cargo install "$NAME" ${VER:+--version "$VER"}
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup $NAME ${VER:-latest}"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
fi
|
|
|
|
|
|
done
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_ok "Setup Rust"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
fi
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs Adminer (Debian/Ubuntu via APT, Alpine via direct download).
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Adds Adminer to Apache or web root
|
|
|
|
|
|
# - Supports Alpine and Debian-based systems
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
2025-06-18 12:07:08 +02:00
|
|
|
|
function setup_adminer() {
|
2025-05-27 14:29:50 +02:00
|
|
|
|
if grep -qi alpine /etc/os-release; then
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Adminer (Alpine)"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
mkdir -p /var/www/localhost/htdocs/adminer
|
|
|
|
|
|
if ! curl -fsSL https://github.com/vrana/adminer/releases/latest/download/adminer.php \
|
|
|
|
|
|
-o /var/www/localhost/htdocs/adminer/index.php; then
|
|
|
|
|
|
msg_error "Failed to download Adminer"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
msg_ok "Adminer available at /adminer (Alpine)"
|
|
|
|
|
|
else
|
2025-06-18 12:07:08 +02:00
|
|
|
|
msg_info "Setup Adminer (Debian/Ubuntu)"
|
2025-05-27 14:29:50 +02:00
|
|
|
|
$STD apt-get install -y adminer
|
|
|
|
|
|
$STD a2enconf adminer
|
|
|
|
|
|
$STD systemctl reload apache2
|
|
|
|
|
|
msg_ok "Adminer available at /adminer (Debian/Ubuntu)"
|
|
|
|
|
|
fi
|
|
|
|
|
|
}
|
2025-06-18 12:07:08 +02:00
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs or updates yq (mikefarah/yq - Go version).
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Checks if yq is installed and from correct source
|
|
|
|
|
|
# - Compares with latest release on GitHub
|
|
|
|
|
|
# - Updates if outdated or wrong implementation
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
function setup_yq() {
|
|
|
|
|
|
local TMP_DIR
|
|
|
|
|
|
TMP_DIR=$(mktemp -d)
|
|
|
|
|
|
local CURRENT_VERSION=""
|
|
|
|
|
|
local BINARY_PATH="/usr/local/bin/yq"
|
|
|
|
|
|
local GITHUB_REPO="mikefarah/yq"
|
|
|
|
|
|
|
|
|
|
|
|
if ! command -v jq &>/dev/null; then
|
|
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y jq || {
|
|
|
|
|
|
msg_error "Failed to install jq"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if command -v yq &>/dev/null; then
|
|
|
|
|
|
if ! yq --version 2>&1 | grep -q 'mikefarah'; then
|
|
|
|
|
|
rm -f "$(command -v yq)"
|
|
|
|
|
|
else
|
|
|
|
|
|
CURRENT_VERSION=$(yq --version | awk '{print $NF}' | sed 's/^v//')
|
|
|
|
|
|
fi
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
local RELEASE_JSON
|
|
|
|
|
|
RELEASE_JSON=$(curl -fsSL "https://api.github.com/repos/${GITHUB_REPO}/releases/latest")
|
|
|
|
|
|
local LATEST_VERSION
|
|
|
|
|
|
LATEST_VERSION=$(echo "$RELEASE_JSON" | jq -r '.tag_name' | sed 's/^v//')
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -z "$LATEST_VERSION" ]]; then
|
|
|
|
|
|
msg_error "Could not determine latest yq version from GitHub."
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -n "$CURRENT_VERSION" && "$CURRENT_VERSION" == "$LATEST_VERSION" ]]; then
|
|
|
|
|
|
return
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
msg_info "Setup yq ($LATEST_VERSION)"
|
|
|
|
|
|
curl -fsSL "https://github.com/${GITHUB_REPO}/releases/download/v${LATEST_VERSION}/yq_linux_amd64" -o "$TMP_DIR/yq"
|
|
|
|
|
|
chmod +x "$TMP_DIR/yq"
|
|
|
|
|
|
mv "$TMP_DIR/yq" "$BINARY_PATH"
|
|
|
|
|
|
|
|
|
|
|
|
if [[ ! -x "$BINARY_PATH" ]]; then
|
|
|
|
|
|
msg_error "Failed to install yq to $BINARY_PATH"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
hash -r
|
|
|
|
|
|
|
|
|
|
|
|
local FINAL_VERSION
|
|
|
|
|
|
FINAL_VERSION=$("$BINARY_PATH" --version 2>/dev/null | awk '{print $NF}')
|
|
|
|
|
|
if [[ "$FINAL_VERSION" == "v$LATEST_VERSION" ]]; then
|
|
|
|
|
|
msg_ok "Setup yq ($LATEST_VERSION)"
|
|
|
|
|
|
else
|
|
|
|
|
|
msg_error "yq installation incomplete or version mismatch"
|
|
|
|
|
|
fi
|
|
|
|
|
|
}
|
2025-06-25 12:37:15 +02:00
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs ImageMagick 7 from source (Debian/Ubuntu only).
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Downloads the latest ImageMagick source tarball
|
|
|
|
|
|
# - Builds and installs ImageMagick to /usr/local
|
|
|
|
|
|
# - Configures dynamic linker (ldconfig)
|
|
|
|
|
|
#
|
|
|
|
|
|
# Notes:
|
|
|
|
|
|
# - Requires: build-essential, libtool, libjpeg-dev, libpng-dev, etc.
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
function setup_imagemagick() {
|
|
|
|
|
|
local TMP_DIR
|
|
|
|
|
|
TMP_DIR=$(mktemp -d)
|
|
|
|
|
|
local VERSION=""
|
|
|
|
|
|
local BINARY_PATH="/usr/local/bin/magick"
|
|
|
|
|
|
|
|
|
|
|
|
if command -v magick &>/dev/null; then
|
|
|
|
|
|
VERSION=$(magick -version | awk '/^Version/ {print $3}')
|
|
|
|
|
|
msg_ok "ImageMagick already installed ($VERSION)"
|
|
|
|
|
|
return 0
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
msg_info "Setup ImageMagick (Patience)"
|
|
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y \
|
|
|
|
|
|
build-essential \
|
|
|
|
|
|
libtool \
|
|
|
|
|
|
libjpeg-dev \
|
|
|
|
|
|
libpng-dev \
|
|
|
|
|
|
libtiff-dev \
|
|
|
|
|
|
libwebp-dev \
|
|
|
|
|
|
libheif-dev \
|
|
|
|
|
|
libde265-dev \
|
|
|
|
|
|
libopenjp2-7-dev \
|
|
|
|
|
|
libxml2-dev \
|
|
|
|
|
|
liblcms2-dev \
|
|
|
|
|
|
libfreetype6-dev \
|
|
|
|
|
|
libraw-dev \
|
|
|
|
|
|
libfftw3-dev \
|
|
|
|
|
|
liblqr-1-0-dev \
|
|
|
|
|
|
libgsl-dev \
|
|
|
|
|
|
pkg-config \
|
|
|
|
|
|
ghostscript
|
|
|
|
|
|
|
|
|
|
|
|
curl -fsSL https://imagemagick.org/archive/ImageMagick.tar.gz -o "$TMP_DIR/ImageMagick.tar.gz"
|
|
|
|
|
|
tar -xzf "$TMP_DIR/ImageMagick.tar.gz" -C "$TMP_DIR"
|
|
|
|
|
|
cd "$TMP_DIR"/ImageMagick-* || {
|
|
|
|
|
|
msg_error "Source extraction failed"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
./configure --disable-static >/dev/null
|
|
|
|
|
|
$STD make
|
|
|
|
|
|
$STD make install
|
|
|
|
|
|
$STD ldconfig /usr/local/lib
|
|
|
|
|
|
|
|
|
|
|
|
if [[ ! -x "$BINARY_PATH" ]]; then
|
|
|
|
|
|
msg_error "ImageMagick installation failed"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
VERSION=$("$BINARY_PATH" -version | awk '/^Version/ {print $3}')
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
ensure_usr_local_bin_persist
|
|
|
|
|
|
msg_ok "Setup ImageMagick $VERSION"
|
|
|
|
|
|
}
|
2025-07-08 17:19:51 +02:00
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
# Installs FFmpeg from source or prebuilt binary (Debian/Ubuntu only).
|
|
|
|
|
|
#
|
|
|
|
|
|
# Description:
|
|
|
|
|
|
# - Downloads and builds FFmpeg from GitHub (https://github.com/FFmpeg/FFmpeg)
|
|
|
|
|
|
# - Supports specific version override via FFMPEG_VERSION (e.g. n7.1.1)
|
|
|
|
|
|
# - Supports build profile via FFMPEG_TYPE:
|
|
|
|
|
|
# - minimal : x264, vpx, mp3 only
|
|
|
|
|
|
# - medium : adds subtitles, fonts, opus, vorbis
|
|
|
|
|
|
# - full : adds dav1d, svt-av1, zlib, numa
|
|
|
|
|
|
# - binary : downloads static build (johnvansickle.com)
|
|
|
|
|
|
# - Defaults to latest stable version and full feature set
|
|
|
|
|
|
#
|
|
|
|
|
|
# Notes:
|
|
|
|
|
|
# - Requires: curl, jq, build-essential, and matching codec libraries
|
|
|
|
|
|
# - Result is installed to /usr/local/bin/ffmpeg
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
function setup_ffmpeg() {
|
|
|
|
|
|
local TMP_DIR
|
|
|
|
|
|
TMP_DIR=$(mktemp -d)
|
|
|
|
|
|
local GITHUB_REPO="FFmpeg/FFmpeg"
|
|
|
|
|
|
local VERSION="${FFMPEG_VERSION:-latest}"
|
|
|
|
|
|
local TYPE="${FFMPEG_TYPE:-full}"
|
|
|
|
|
|
local BIN_PATH="/usr/local/bin/ffmpeg"
|
|
|
|
|
|
|
|
|
|
|
|
# Binary fallback mode
|
|
|
|
|
|
if [[ "$TYPE" == "binary" ]]; then
|
|
|
|
|
|
msg_info "Installing FFmpeg (static binary)"
|
|
|
|
|
|
curl -fsSL https://johnvansickle.com/ffmpeg/releases/ffmpeg-release-amd64-static.tar.xz -o "$TMP_DIR/ffmpeg.tar.xz"
|
|
|
|
|
|
tar -xf "$TMP_DIR/ffmpeg.tar.xz" -C "$TMP_DIR"
|
|
|
|
|
|
local EXTRACTED_DIR
|
|
|
|
|
|
EXTRACTED_DIR=$(find "$TMP_DIR" -maxdepth 1 -type d -name "ffmpeg-*")
|
|
|
|
|
|
cp "$EXTRACTED_DIR/ffmpeg" "$BIN_PATH"
|
|
|
|
|
|
cp "$EXTRACTED_DIR/ffprobe" /usr/local/bin/ffprobe
|
|
|
|
|
|
chmod +x "$BIN_PATH" /usr/local/bin/ffprobe
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
msg_ok "Installed FFmpeg binary ($($BIN_PATH -version | head -n1))"
|
|
|
|
|
|
return
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if ! command -v jq &>/dev/null; then
|
|
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y jq
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
# Auto-detect latest stable version if none specified
|
|
|
|
|
|
if [[ "$VERSION" == "latest" || -z "$VERSION" ]]; then
|
|
|
|
|
|
msg_info "Resolving latest FFmpeg tag"
|
|
|
|
|
|
VERSION=$(curl -fsSL "https://api.github.com/repos/${GITHUB_REPO}/tags" |
|
|
|
|
|
|
jq -r '.[].name' |
|
|
|
|
|
|
grep -E '^n[0-9]+\.[0-9]+\.[0-9]+$' |
|
|
|
|
|
|
sort -V | tail -n1)
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ -z "$VERSION" ]]; then
|
|
|
|
|
|
msg_error "Could not determine FFmpeg version"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
msg_info "Installing FFmpeg ${VERSION} ($TYPE)"
|
|
|
|
|
|
|
|
|
|
|
|
# Dependency selection
|
|
|
|
|
|
local DEPS=(build-essential yasm nasm pkg-config)
|
|
|
|
|
|
case "$TYPE" in
|
|
|
|
|
|
minimal)
|
|
|
|
|
|
DEPS+=(libx264-dev libvpx-dev libmp3lame-dev)
|
|
|
|
|
|
;;
|
|
|
|
|
|
medium)
|
|
|
|
|
|
DEPS+=(libx264-dev libvpx-dev libmp3lame-dev libfreetype6-dev libass-dev libopus-dev libvorbis-dev)
|
|
|
|
|
|
;;
|
|
|
|
|
|
full)
|
|
|
|
|
|
DEPS+=(
|
|
|
|
|
|
libx264-dev libx265-dev libvpx-dev libmp3lame-dev
|
|
|
|
|
|
libfreetype6-dev libass-dev libopus-dev libvorbis-dev
|
|
|
|
|
|
libdav1d-dev libsvtav1-dev zlib1g-dev libnuma-dev
|
2025-07-15 16:57:43 +02:00
|
|
|
|
libva-dev libdrm-dev
|
2025-07-08 17:19:51 +02:00
|
|
|
|
)
|
|
|
|
|
|
;;
|
|
|
|
|
|
*)
|
|
|
|
|
|
msg_error "Invalid FFMPEG_TYPE: $TYPE"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
;;
|
|
|
|
|
|
esac
|
|
|
|
|
|
|
|
|
|
|
|
$STD apt-get update
|
|
|
|
|
|
$STD apt-get install -y "${DEPS[@]}"
|
|
|
|
|
|
|
|
|
|
|
|
curl -fsSL "https://github.com/${GITHUB_REPO}/archive/refs/tags/${VERSION}.tar.gz" -o "$TMP_DIR/ffmpeg.tar.gz"
|
|
|
|
|
|
tar -xzf "$TMP_DIR/ffmpeg.tar.gz" -C "$TMP_DIR"
|
|
|
|
|
|
cd "$TMP_DIR/FFmpeg-"* || {
|
|
|
|
|
|
msg_error "Source extraction failed"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
local args=(
|
|
|
|
|
|
--enable-gpl
|
|
|
|
|
|
--enable-shared
|
|
|
|
|
|
--enable-nonfree
|
|
|
|
|
|
--disable-static
|
|
|
|
|
|
--enable-libx264
|
|
|
|
|
|
--enable-libvpx
|
|
|
|
|
|
--enable-libmp3lame
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$TYPE" != "minimal" ]]; then
|
|
|
|
|
|
args+=(--enable-libfreetype --enable-libass --enable-libopus --enable-libvorbis)
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ "$TYPE" == "full" ]]; then
|
|
|
|
|
|
args+=(--enable-libx265 --enable-libdav1d --enable-zlib)
|
2025-07-15 16:57:43 +02:00
|
|
|
|
args+=(--enable-vaapi --enable-libdrm)
|
2025-07-08 17:19:51 +02:00
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
if [[ ${#args[@]} -eq 0 ]]; then
|
|
|
|
|
|
msg_error "FFmpeg configure args array is empty – aborting."
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
./configure "${args[@]}" >"$TMP_DIR/configure.log" 2>&1 || {
|
|
|
|
|
|
msg_error "FFmpeg ./configure failed (see $TMP_DIR/configure.log)"
|
|
|
|
|
|
cat "$TMP_DIR/configure.log" | tail -n 20
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
$STD make -j"$(nproc)"
|
|
|
|
|
|
$STD make install
|
|
|
|
|
|
echo "/usr/local/lib" >/etc/ld.so.conf.d/ffmpeg.conf
|
|
|
|
|
|
ldconfig
|
|
|
|
|
|
|
|
|
|
|
|
ldconfig -p | grep libavdevice >/dev/null || {
|
|
|
|
|
|
msg_error "libavdevice not registered with dynamic linker"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if ! command -v ffmpeg &>/dev/null; then
|
|
|
|
|
|
msg_error "FFmpeg installation failed"
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
|
|
local FINAL_VERSION
|
|
|
|
|
|
FINAL_VERSION=$(ffmpeg -version | head -n1 | awk '{print $3}')
|
|
|
|
|
|
rm -rf "$TMP_DIR"
|
|
|
|
|
|
ensure_usr_local_bin_persist
|
|
|
|
|
|
msg_ok "Setup FFmpeg $FINAL_VERSION"
|
|
|
|
|
|
}
|