mirror of
https://github.com/community-scripts/ProxmoxVE.git
synced 2026-05-17 05:54:56 +02:00
Compare commits
2 Commits
bump-node
...
add-script
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5141a58de4 | ||
|
|
09bc0abcad |
35
CHANGELOG.md
35
CHANGELOG.md
@@ -461,37 +461,6 @@ Exercise vigilance regarding copycat or coat-tailing sites that seek to exploit
|
|||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
## 2026-05-12
|
|
||||||
|
|
||||||
### 🚀 Updated Scripts
|
|
||||||
|
|
||||||
- #### 🐞 Bug Fixes
|
|
||||||
|
|
||||||
- Save Omada version [@lucacome](https://github.com/lucacome) ([#14433](https://github.com/community-scripts/ProxmoxVE/pull/14433))
|
|
||||||
|
|
||||||
## 2026-05-11
|
|
||||||
|
|
||||||
### 🆕 New Scripts
|
|
||||||
|
|
||||||
- Lychee ([#14424](https://github.com/community-scripts/ProxmoxVE/pull/14424))
|
|
||||||
|
|
||||||
### 🚀 Updated Scripts
|
|
||||||
|
|
||||||
- #### 🐞 Bug Fixes
|
|
||||||
|
|
||||||
- Termix: fix nginx pid path and log paths on update (#) [@MickLesk](https://github.com/MickLesk) ([#14419](https://github.com/community-scripts/ProxmoxVE/pull/14419))
|
|
||||||
- Nginxproxymanager: restore NPM nginx.conf after OpenResty rebuid [@MickLesk](https://github.com/MickLesk) ([#14421](https://github.com/community-scripts/ProxmoxVE/pull/14421))
|
|
||||||
|
|
||||||
- #### 🔧 Refactor
|
|
||||||
|
|
||||||
- InvestBrain: add commented reverse proxy config hints to .env [@MickLesk](https://github.com/MickLesk) ([#14422](https://github.com/community-scripts/ProxmoxVE/pull/14422))
|
|
||||||
|
|
||||||
### 🧰 Tools
|
|
||||||
|
|
||||||
- #### 🐞 Bug Fixes
|
|
||||||
|
|
||||||
- Cronmaster: fix unexpected EOF in update_cronmaster script [@MickLesk](https://github.com/MickLesk) ([#14420](https://github.com/community-scripts/ProxmoxVE/pull/14420))
|
|
||||||
|
|
||||||
## 2026-05-10
|
## 2026-05-10
|
||||||
|
|
||||||
### 🚀 Updated Scripts
|
### 🚀 Updated Scripts
|
||||||
@@ -502,10 +471,6 @@ Exercise vigilance regarding copycat or coat-tailing sites that seek to exploit
|
|||||||
- karakeep: Fix SERVER_VERSION update [@MickLesk](https://github.com/MickLesk) ([#14378](https://github.com/community-scripts/ProxmoxVE/pull/14378))
|
- karakeep: Fix SERVER_VERSION update [@MickLesk](https://github.com/MickLesk) ([#14378](https://github.com/community-scripts/ProxmoxVE/pull/14378))
|
||||||
- inspIRCd: Fix service not autostarting [@tremor021](https://github.com/tremor021) ([#14368](https://github.com/community-scripts/ProxmoxVE/pull/14368))
|
- inspIRCd: Fix service not autostarting [@tremor021](https://github.com/tremor021) ([#14368](https://github.com/community-scripts/ProxmoxVE/pull/14368))
|
||||||
|
|
||||||
- #### 🔧 Refactor
|
|
||||||
|
|
||||||
- refactor: webcheck [@CrazyWolf13](https://github.com/CrazyWolf13) ([#14391](https://github.com/community-scripts/ProxmoxVE/pull/14391))
|
|
||||||
|
|
||||||
### 💾 Core
|
### 💾 Core
|
||||||
|
|
||||||
- #### 🐞 Bug Fixes
|
- #### 🐞 Bug Fixes
|
||||||
|
|||||||
@@ -28,9 +28,6 @@ function update_script() {
|
|||||||
msg_error "No ${APP} Installation Found!"
|
msg_error "No ${APP} Installation Found!"
|
||||||
exit
|
exit
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NODE_VERSION="24" NODE_MODULE="yarn" setup_nodejs
|
|
||||||
|
|
||||||
if check_for_gh_release "excalidraw" "excalidraw/excalidraw"; then
|
if check_for_gh_release "excalidraw" "excalidraw/excalidraw"; then
|
||||||
msg_info "Stopping Service"
|
msg_info "Stopping Service"
|
||||||
systemctl stop excalidraw
|
systemctl stop excalidraw
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
__ __
|
|
||||||
/ / __ _______/ /_ ___ ___
|
|
||||||
/ / / / / / ___/ __ \/ _ \/ _ \
|
|
||||||
/ /___/ /_/ / /__/ / / / __/ __/
|
|
||||||
/_____/\__, /\___/_/ /_/\___/\___/
|
|
||||||
/____/
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
_____ __ ___ __
|
|
||||||
/ ___// /_ / (_)___ / /__
|
|
||||||
\__ \/ __ \/ / / __ \/ //_/
|
|
||||||
___/ / / / / / / / / / ,<
|
|
||||||
/____/_/ /_/_/_/_/ /_/_/|_|
|
|
||||||
|
|
||||||
73
ct/lychee.sh
73
ct/lychee.sh
@@ -1,73 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
source <(curl -fsSL https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/misc/build.func)
|
|
||||||
# Copyright (c) 2021-2026 community-scripts ORG
|
|
||||||
# Author: MickLesk (CanbiZ)
|
|
||||||
# License: MIT | https://github.com/community-scripts/ProxmoxVE/raw/main/LICENSE
|
|
||||||
# Source: https://github.com/LycheeOrg/Lychee
|
|
||||||
|
|
||||||
APP="Lychee"
|
|
||||||
var_tags="${var_tags:-media;photos;gallery}"
|
|
||||||
var_cpu="${var_cpu:-2}"
|
|
||||||
var_ram="${var_ram:-2048}"
|
|
||||||
var_disk="${var_disk:-8}"
|
|
||||||
var_os="${var_os:-debian}"
|
|
||||||
var_version="${var_version:-13}"
|
|
||||||
var_unprivileged="${var_unprivileged:-1}"
|
|
||||||
|
|
||||||
header_info "$APP"
|
|
||||||
variables
|
|
||||||
color
|
|
||||||
catch_errors
|
|
||||||
|
|
||||||
function update_script() {
|
|
||||||
header_info
|
|
||||||
check_container_storage
|
|
||||||
check_container_resources
|
|
||||||
|
|
||||||
if [[ ! -d /opt/lychee ]]; then
|
|
||||||
msg_error "No ${APP} Installation Found!"
|
|
||||||
exit
|
|
||||||
fi
|
|
||||||
|
|
||||||
if check_for_gh_release "lychee" "LycheeOrg/Lychee"; then
|
|
||||||
msg_info "Stopping Services"
|
|
||||||
systemctl stop caddy
|
|
||||||
msg_ok "Stopped Services"
|
|
||||||
|
|
||||||
msg_info "Backing up Data"
|
|
||||||
cp /opt/lychee/.env /opt/lychee.env.bak
|
|
||||||
cp -r /opt/lychee/storage /opt/lychee_storage_backup
|
|
||||||
msg_ok "Backed up Data"
|
|
||||||
|
|
||||||
CLEAN_INSTALL=1 fetch_and_deploy_gh_release "lychee" "LycheeOrg/Lychee" "prebuild" "latest" "/opt/lychee" "Lychee.zip"
|
|
||||||
|
|
||||||
msg_info "Restoring Data"
|
|
||||||
cp /opt/lychee.env.bak /opt/lychee/.env
|
|
||||||
rm -f /opt/lychee.env.bak
|
|
||||||
cp -r /opt/lychee_storage_backup/. /opt/lychee/storage
|
|
||||||
rm -rf /opt/lychee_storage_backup
|
|
||||||
msg_ok "Restored Data"
|
|
||||||
|
|
||||||
msg_info "Updating Application"
|
|
||||||
cd /opt/lychee
|
|
||||||
$STD php artisan migrate --force
|
|
||||||
$STD php artisan optimize:clear
|
|
||||||
chmod -R 775 /opt/lychee/storage /opt/lychee/bootstrap/cache
|
|
||||||
msg_ok "Updated Application"
|
|
||||||
|
|
||||||
msg_info "Starting Services"
|
|
||||||
systemctl start caddy
|
|
||||||
msg_ok "Started Services"
|
|
||||||
msg_ok "Updated successfully!"
|
|
||||||
fi
|
|
||||||
exit
|
|
||||||
}
|
|
||||||
|
|
||||||
start
|
|
||||||
build_container
|
|
||||||
description
|
|
||||||
|
|
||||||
msg_ok "Completed Successfully!\n"
|
|
||||||
echo -e "${CREATING}${GN}${APP} setup has been successfully initialized!${CL}"
|
|
||||||
echo -e "${INFO}${YW} Access it using the following URL:${CL}"
|
|
||||||
echo -e "${TAB}${GATEWAY}${BGN}http://${IP}${CL}"
|
|
||||||
@@ -92,11 +92,6 @@ ExecStart=/usr/local/openresty/nginx/sbin/nginx -g 'daemon off;'
|
|||||||
[Install]
|
[Install]
|
||||||
WantedBy=multi-user.target
|
WantedBy=multi-user.target
|
||||||
EOF
|
EOF
|
||||||
if [ -f /opt/nginxproxymanager/docker/rootfs/etc/nginx/nginx.conf ]; then
|
|
||||||
cp /opt/nginxproxymanager/docker/rootfs/etc/nginx/nginx.conf /usr/local/openresty/nginx/conf/nginx.conf
|
|
||||||
sed -i 's+^daemon+#daemon+g' /usr/local/openresty/nginx/conf/nginx.conf
|
|
||||||
sed -i 's+include conf.d+include /etc/nginx/conf.d+g' /usr/local/openresty/nginx/conf/nginx.conf
|
|
||||||
fi
|
|
||||||
sed -i 's/user npm/user root/g; s/^pid/#pid/g' /usr/local/openresty/nginx/conf/nginx.conf
|
sed -i 's/user npm/user root/g; s/^pid/#pid/g' /usr/local/openresty/nginx/conf/nginx.conf
|
||||||
systemctl daemon-reload
|
systemctl daemon-reload
|
||||||
systemctl unmask openresty 2>/dev/null || true
|
systemctl unmask openresty 2>/dev/null || true
|
||||||
|
|||||||
32
ct/omada.sh
32
ct/omada.sh
@@ -38,32 +38,20 @@ function update_script() {
|
|||||||
|
|
||||||
JAVA_VERSION="21" setup_java
|
JAVA_VERSION="21" setup_java
|
||||||
|
|
||||||
|
msg_info "Updating Omada Controller"
|
||||||
OMADA_URL=$(curl -fsSL "https://support.omadanetworks.com/en/download/software/omada-controller/" |
|
OMADA_URL=$(curl -fsSL "https://support.omadanetworks.com/en/download/software/omada-controller/" |
|
||||||
grep -o 'https://static\.tp-link\.com/upload/software/[^"]*linux_x64[^"]*\.deb' |
|
grep -o 'https://static\.tp-link\.com/upload/software/[^"]*linux_x64[^"]*\.deb' |
|
||||||
head -n1)
|
head -n1)
|
||||||
OMADA_PKG=$(basename "${OMADA_URL}")
|
OMADA_PKG=$(basename "$OMADA_URL")
|
||||||
VERSION=$(sed -n 's/.*_v\([0-9.]*\)_.*_\([0-9]\{14\}\)\.deb$/\1-\2/p' <<<"${OMADA_PKG}")
|
if [ -z "$OMADA_PKG" ]; then
|
||||||
|
msg_error "Could not retrieve Omada package – server may be down."
|
||||||
CURRENT_VERSION=$(cat $HOME/.omada 2>/dev/null || echo "0")
|
exit
|
||||||
|
|
||||||
if dpkg --compare-versions "${VERSION}" gt "${CURRENT_VERSION}"; then
|
|
||||||
|
|
||||||
msg_info "Updating Omada Controller"
|
|
||||||
|
|
||||||
if [ -z "${OMADA_PKG}" ]; then
|
|
||||||
msg_error "Could not retrieve Omada package – server may be down."
|
|
||||||
exit
|
|
||||||
fi
|
|
||||||
curl -fsSL "${OMADA_URL}" -o "${OMADA_PKG}"
|
|
||||||
export DEBIAN_FRONTEND=noninteractive
|
|
||||||
$STD dpkg -i "${OMADA_PKG}"
|
|
||||||
rm -f "${OMADA_PKG}"
|
|
||||||
echo "${VERSION}" >$HOME/.omada
|
|
||||||
msg_ok "Updated Omada Controller to ${VERSION}"
|
|
||||||
msg_ok "Updated successfully!"
|
|
||||||
else
|
|
||||||
msg_ok "No update available: ${APP} (${CURRENT_VERSION})"
|
|
||||||
fi
|
fi
|
||||||
|
curl -fsSL "$OMADA_URL" -o "$OMADA_PKG"
|
||||||
|
export DEBIAN_FRONTEND=noninteractive
|
||||||
|
$STD dpkg -i "$OMADA_PKG"
|
||||||
|
rm -f "$OMADA_PKG"
|
||||||
|
msg_ok "Updated successfully!"
|
||||||
exit
|
exit
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
85
ct/shlink.sh
85
ct/shlink.sh
@@ -1,85 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
source <(curl -fsSL https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/misc/build.func)
|
|
||||||
# Copyright (c) 2021-2026 community-scripts ORG
|
|
||||||
# Author: MickLesk (CanbiZ)
|
|
||||||
# License: MIT | https://github.com/community-scripts/ProxmoxVE/raw/main/LICENSE
|
|
||||||
# Source: https://shlink.io/
|
|
||||||
|
|
||||||
APP="Shlink"
|
|
||||||
var_tags="${var_tags:-url-shortener;analytics;php}"
|
|
||||||
var_cpu="${var_cpu:-2}"
|
|
||||||
var_ram="${var_ram:-2048}"
|
|
||||||
var_disk="${var_disk:-4}"
|
|
||||||
var_os="${var_os:-debian}"
|
|
||||||
var_version="${var_version:-13}"
|
|
||||||
var_unprivileged="${var_unprivileged:-1}"
|
|
||||||
|
|
||||||
header_info "$APP"
|
|
||||||
variables
|
|
||||||
color
|
|
||||||
catch_errors
|
|
||||||
|
|
||||||
function update_script() {
|
|
||||||
header_info
|
|
||||||
check_container_storage
|
|
||||||
check_container_resources
|
|
||||||
|
|
||||||
if [[ ! -d /opt/shlink ]]; then
|
|
||||||
msg_error "No ${APP} Installation Found!"
|
|
||||||
exit
|
|
||||||
fi
|
|
||||||
|
|
||||||
if check_for_gh_release "shlink" "shlinkio/shlink"; then
|
|
||||||
msg_info "Stopping Service"
|
|
||||||
systemctl stop shlink
|
|
||||||
msg_ok "Stopped Service"
|
|
||||||
|
|
||||||
msg_info "Backing up Data"
|
|
||||||
cp /opt/shlink/.env /opt/shlink.env.bak
|
|
||||||
cp -r /opt/shlink/data /opt/shlink_data_backup
|
|
||||||
msg_ok "Backed up Data"
|
|
||||||
|
|
||||||
CLEAN_INSTALL=1 fetch_and_deploy_gh_release "shlink" "shlinkio/shlink" "prebuild" "latest" "/opt/shlink" "shlink*_php8.5_dist.zip"
|
|
||||||
|
|
||||||
msg_info "Restoring Data"
|
|
||||||
cp /opt/shlink.env.bak /opt/shlink/.env
|
|
||||||
rm -f /opt/shlink.env.bak
|
|
||||||
cp -r /opt/shlink_data_backup/. /opt/shlink/data
|
|
||||||
rm -rf /opt/shlink_data_backup
|
|
||||||
msg_ok "Restored Data"
|
|
||||||
|
|
||||||
msg_info "Updating Application"
|
|
||||||
cd /opt/shlink
|
|
||||||
$STD php ./vendor/bin/rr get --no-interaction --location bin/
|
|
||||||
chmod +x bin/rr
|
|
||||||
set -a
|
|
||||||
source /opt/shlink/.env
|
|
||||||
set +a
|
|
||||||
$STD php vendor/bin/shlink-installer init --no-interaction --clear-db-cache --skip-download-geolite
|
|
||||||
msg_ok "Updated Application"
|
|
||||||
|
|
||||||
msg_info "Starting Service"
|
|
||||||
systemctl start shlink
|
|
||||||
msg_ok "Started Service"
|
|
||||||
msg_ok "Updated successfully!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -d /opt/shlink-web-client ]]; then
|
|
||||||
if check_for_gh_release "shlink-web-client" "shlinkio/shlink-web-client"; then
|
|
||||||
CLEAN_INSTALL=1 fetch_and_deploy_gh_release "shlink-web-client" "shlinkio/shlink-web-client" "prebuild" "latest" "/opt/shlink-web-client" "shlink-web-client_*_dist.zip"
|
|
||||||
msg_ok "Updated Web Client"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
exit
|
|
||||||
}
|
|
||||||
|
|
||||||
start
|
|
||||||
build_container
|
|
||||||
description
|
|
||||||
|
|
||||||
msg_ok "Completed Successfully!\n"
|
|
||||||
echo -e "${CREATING}${GN}${APP} setup has been successfully initialized!${CL}"
|
|
||||||
echo -e "${INFO}${YW} Access Shlink Web Client using the following URL:${CL}"
|
|
||||||
echo -e "${TAB}${GATEWAY}${BGN}http://${IP}:3000${CL}"
|
|
||||||
echo -e "${INFO}${YW} Shlink HTTP API:${CL}"
|
|
||||||
echo -e "${TAB}${GATEWAY}${BGN}http://${IP}:8080${CL}"
|
|
||||||
@@ -29,8 +29,6 @@ function update_script() {
|
|||||||
exit
|
exit
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NODE_VERSION="24" setup_nodejs
|
|
||||||
|
|
||||||
if check_for_gh_tag "guacd" "apache/guacamole-server"; then
|
if check_for_gh_tag "guacd" "apache/guacamole-server"; then
|
||||||
msg_info "Stopping guacd"
|
msg_info "Stopping guacd"
|
||||||
systemctl stop guacd 2>/dev/null || true
|
systemctl stop guacd 2>/dev/null || true
|
||||||
@@ -157,6 +155,8 @@ EOF
|
|||||||
/opt/termix/nginx/client_body
|
/opt/termix/nginx/client_body
|
||||||
msg_ok "Recreated Directories"
|
msg_ok "Recreated Directories"
|
||||||
|
|
||||||
|
NODE_VERSION="24" setup_nodejs
|
||||||
|
|
||||||
msg_info "Building Frontend"
|
msg_info "Building Frontend"
|
||||||
cd /opt/termix
|
cd /opt/termix
|
||||||
export COREPACK_ENABLE_DOWNLOAD_PROMPT=0
|
export COREPACK_ENABLE_DOWNLOAD_PROMPT=0
|
||||||
@@ -199,9 +199,7 @@ EOF
|
|||||||
cp /etc/nginx/nginx.conf /etc/nginx/nginx.conf.bak
|
cp /etc/nginx/nginx.conf /etc/nginx/nginx.conf.bak
|
||||||
curl -fsSL "https://raw.githubusercontent.com/Termix-SSH/Termix/main/docker/nginx.conf" -o /etc/nginx/nginx.conf
|
curl -fsSL "https://raw.githubusercontent.com/Termix-SSH/Termix/main/docker/nginx.conf" -o /etc/nginx/nginx.conf
|
||||||
sed -i '/^master_process/d' /etc/nginx/nginx.conf
|
sed -i '/^master_process/d' /etc/nginx/nginx.conf
|
||||||
sed -i 's|pid /tmp/nginx/nginx.pid;|pid /run/nginx.pid;|' /etc/nginx/nginx.conf
|
sed -i '/^pid \/app\/nginx/d' /etc/nginx/nginx.conf
|
||||||
sed -i 's|error_log /tmp/nginx/error.log|error_log /var/log/nginx/error.log|' /etc/nginx/nginx.conf
|
|
||||||
sed -i 's|access_log /tmp/nginx/access.log|access_log /var/log/nginx/access.log|' /etc/nginx/nginx.conf
|
|
||||||
sed -i 's|/app/html|/opt/termix/html|g' /etc/nginx/nginx.conf
|
sed -i 's|/app/html|/opt/termix/html|g' /etc/nginx/nginx.conf
|
||||||
sed -i 's|/app/nginx|/opt/termix/nginx|g' /etc/nginx/nginx.conf
|
sed -i 's|/app/nginx|/opt/termix/nginx|g' /etc/nginx/nginx.conf
|
||||||
sed -i 's|listen ${PORT};|listen 80;|g' /etc/nginx/nginx.conf
|
sed -i 's|listen ${PORT};|listen 80;|g' /etc/nginx/nginx.conf
|
||||||
|
|||||||
@@ -29,8 +29,6 @@ function update_script() {
|
|||||||
exit
|
exit
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NODE_VERSION="24" setup_nodejs
|
|
||||||
|
|
||||||
if check_for_gh_release "trek" "mauriceboe/TREK"; then
|
if check_for_gh_release "trek" "mauriceboe/TREK"; then
|
||||||
msg_info "Stopping Service"
|
msg_info "Stopping Service"
|
||||||
systemctl stop trek
|
systemctl stop trek
|
||||||
|
|||||||
@@ -28,10 +28,7 @@ function update_script() {
|
|||||||
msg_error "No ${APP} Installation Found!"
|
msg_error "No ${APP} Installation Found!"
|
||||||
exit
|
exit
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NODE_VERSION="24" setup_nodejs
|
|
||||||
setup_mariadb
|
setup_mariadb
|
||||||
|
|
||||||
if check_for_gh_release "wallabag" "wallabag/wallabag"; then
|
if check_for_gh_release "wallabag" "wallabag/wallabag"; then
|
||||||
msg_info "Stopping Services"
|
msg_info "Stopping Services"
|
||||||
systemctl stop nginx php8.3-fpm
|
systemctl stop nginx php8.3-fpm
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ function update_script() {
|
|||||||
exit
|
exit
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if check_for_gh_release "web-check" "Lissy93/web-check"; then
|
if check_for_gh_release "web-check" "CrazyWolf13/web-check"; then
|
||||||
msg_info "Stopping Service"
|
msg_info "Stopping Service"
|
||||||
systemctl stop web-check
|
systemctl stop web-check
|
||||||
msg_ok "Stopped Service"
|
msg_ok "Stopped Service"
|
||||||
@@ -38,7 +38,7 @@ function update_script() {
|
|||||||
msg_ok "Created backup"
|
msg_ok "Created backup"
|
||||||
|
|
||||||
NODE_VERSION="22" NODE_MODULE="yarn" setup_nodejs
|
NODE_VERSION="22" NODE_MODULE="yarn" setup_nodejs
|
||||||
CLEAN_INSTALL=1 fetch_and_deploy_gh_release "web-check" "Lissy93/web-check" "tarball"
|
CLEAN_INSTALL=1 fetch_and_deploy_gh_release "web-check" "CrazyWolf13/web-check" "tarball"
|
||||||
|
|
||||||
msg_info "Restoring backup"
|
msg_info "Restoring backup"
|
||||||
mv /opt/.env /opt/web-check
|
mv /opt/.env /opt/web-check
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ function update_script() {
|
|||||||
exit
|
exit
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NODE_VERSION="24" NODE_MODULE="yarn,node-gyp" setup_nodejs
|
NODE_VERSION="22" NODE_MODULE="yarn,node-gyp" setup_nodejs
|
||||||
|
|
||||||
if check_for_gh_release "wikijs" "requarks/wiki"; then
|
if check_for_gh_release "wikijs" "requarks/wiki"; then
|
||||||
msg_info "Verifying whether ${APP}' new release is v3.x+ and current install uses SQLite."
|
msg_info "Verifying whether ${APP}' new release is v3.x+ and current install uses SQLite."
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ msg_info "Installing Dependencies"
|
|||||||
$STD apt install -y xdg-utils
|
$STD apt install -y xdg-utils
|
||||||
msg_ok "Installed Dependencies"
|
msg_ok "Installed Dependencies"
|
||||||
|
|
||||||
NODE_VERSION="24" NODE_MODULE="yarn" setup_nodejs
|
NODE_VERSION="22" NODE_MODULE="yarn" setup_nodejs
|
||||||
fetch_and_deploy_gh_release "excalidraw" "excalidraw/excalidraw" "tarball"
|
fetch_and_deploy_gh_release "excalidraw" "excalidraw/excalidraw" "tarball"
|
||||||
|
|
||||||
msg_info "Configuring Excalidraw"
|
msg_info "Configuring Excalidraw"
|
||||||
|
|||||||
@@ -90,11 +90,6 @@ MAIL_PORT=2525
|
|||||||
MAIL_FROM_ADDRESS="investbrain@${LOCAL_IP}"
|
MAIL_FROM_ADDRESS="investbrain@${LOCAL_IP}"
|
||||||
|
|
||||||
VITE_APP_NAME=Investbrain
|
VITE_APP_NAME=Investbrain
|
||||||
|
|
||||||
# Reverse Proxy Support (uncomment and set APP_URL/ASSET_URL to your domain when using a reverse proxy)
|
|
||||||
# APP_URL=https://your-domain.com
|
|
||||||
# ASSET_URL=https://your-domain.com
|
|
||||||
# TRUSTED_PROXIES=*
|
|
||||||
EOF
|
EOF
|
||||||
export COMPOSER_ALLOW_SUPERUSER=1
|
export COMPOSER_ALLOW_SUPERUSER=1
|
||||||
$STD /usr/local/bin/composer install --no-interaction --no-dev --optimize-autoloader
|
$STD /usr/local/bin/composer install --no-interaction --no-dev --optimize-autoloader
|
||||||
|
|||||||
@@ -1,75 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
# Copyright (c) 2021-2026 community-scripts ORG
|
|
||||||
# Author: MickLesk (CanbiZ)
|
|
||||||
# License: MIT | https://github.com/community-scripts/ProxmoxVE/raw/main/LICENSE
|
|
||||||
# Source: https://github.com/LycheeOrg/Lychee
|
|
||||||
|
|
||||||
source /dev/stdin <<<"$FUNCTIONS_FILE_PATH"
|
|
||||||
color
|
|
||||||
verb_ip6
|
|
||||||
catch_errors
|
|
||||||
setting_up_container
|
|
||||||
network_check
|
|
||||||
update_os
|
|
||||||
|
|
||||||
msg_info "Installing Dependencies"
|
|
||||||
$STD apt install -y \
|
|
||||||
caddy \
|
|
||||||
libimage-exiftool-perl \
|
|
||||||
jpegoptim
|
|
||||||
msg_ok "Installed Dependencies"
|
|
||||||
|
|
||||||
PHP_VERSION="8.4" PHP_FPM="YES" PHP_MODULE="bcmath,ldap,exif,gd,intl,imagick,redis,zip,pdo_pgsql,pcntl" setup_php
|
|
||||||
PG_VERSION="16" setup_postgresql
|
|
||||||
PG_DB_NAME="lychee" PG_DB_USER="lychee" setup_postgresql_db
|
|
||||||
setup_ffmpeg
|
|
||||||
setup_imagemagick
|
|
||||||
|
|
||||||
fetch_and_deploy_gh_release "lychee" "LycheeOrg/Lychee" "prebuild" "latest" "/opt/lychee" "Lychee.zip"
|
|
||||||
|
|
||||||
msg_info "Configuring Application"
|
|
||||||
cd /opt/lychee
|
|
||||||
cp .env.example .env
|
|
||||||
APP_KEY=$($STD php artisan key:generate --show)
|
|
||||||
sed -i "s|^APP_KEY=.*|APP_KEY=${APP_KEY}|" .env
|
|
||||||
sed -i "s|^APP_ENV=.*|APP_ENV=production|" .env
|
|
||||||
sed -i "s|^APP_DEBUG=.*|APP_DEBUG=false|" .env
|
|
||||||
sed -i "s|^APP_URL=.*|APP_URL=http://${LOCAL_IP}|" .env
|
|
||||||
sed -i "s|^DB_CONNECTION=.*|DB_CONNECTION=pgsql|" .env
|
|
||||||
sed -i "s|^DB_HOST=.*|DB_HOST=127.0.0.1|" .env
|
|
||||||
sed -i "s|^DB_PORT=.*|DB_PORT=5432|" .env
|
|
||||||
sed -i "s|^#\?DB_DATABASE=.*|DB_DATABASE=${PG_DB_NAME}|" .env
|
|
||||||
sed -i "s|^DB_USERNAME=.*|DB_USERNAME=${PG_DB_USER}|" .env
|
|
||||||
sed -i "s|^DB_PASSWORD=.*|DB_PASSWORD=${PG_DB_PASS}|" .env
|
|
||||||
mkdir -p storage/framework/{cache,sessions,views} storage/logs bootstrap/cache public/dist public/uploads public/sym
|
|
||||||
touch public/dist/user.css public/dist/custom.js
|
|
||||||
chmod -R 775 storage bootstrap/cache public/dist public/uploads public/sym
|
|
||||||
msg_ok "Configured Application"
|
|
||||||
|
|
||||||
msg_info "Running Database Migrations"
|
|
||||||
cd /opt/lychee
|
|
||||||
$STD php artisan migrate --force
|
|
||||||
msg_ok "Ran Database Migrations"
|
|
||||||
|
|
||||||
chown -R www-data:www-data /opt/lychee
|
|
||||||
|
|
||||||
msg_info "Configuring Caddy"
|
|
||||||
PHP_VER=$(php -r 'echo PHP_MAJOR_VERSION . "." . PHP_MINOR_VERSION;')
|
|
||||||
cat <<EOF >/etc/caddy/Caddyfile
|
|
||||||
:80 {
|
|
||||||
root * /opt/lychee/public
|
|
||||||
php_fastcgi unix//run/php/php${PHP_VER}-fpm.sock
|
|
||||||
file_server
|
|
||||||
encode gzip
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
usermod -aG www-data caddy
|
|
||||||
msg_ok "Configured Caddy"
|
|
||||||
|
|
||||||
systemctl enable -q --now php${PHP_VER}-fpm
|
|
||||||
systemctl restart caddy
|
|
||||||
|
|
||||||
motd_ssh
|
|
||||||
customize
|
|
||||||
cleanup_lxc
|
|
||||||
@@ -38,12 +38,10 @@ msg_info "Installing Omada Controller"
|
|||||||
OMADA_URL=$(curl -fsSL "https://support.omadanetworks.com/en/download/software/omada-controller/" |
|
OMADA_URL=$(curl -fsSL "https://support.omadanetworks.com/en/download/software/omada-controller/" |
|
||||||
grep -o 'https://static\.tp-link\.com/upload/software/[^"]*linux_x64[^"]*\.deb' |
|
grep -o 'https://static\.tp-link\.com/upload/software/[^"]*linux_x64[^"]*\.deb' |
|
||||||
head -n1)
|
head -n1)
|
||||||
OMADA_PKG=$(basename "${OMADA_URL}")
|
OMADA_PKG=$(basename "$OMADA_URL")
|
||||||
curl -fsSL "${OMADA_URL}" -o "${OMADA_PKG}"
|
curl -fsSL "$OMADA_URL" -o "$OMADA_PKG"
|
||||||
$STD dpkg -i "${OMADA_PKG}"
|
$STD dpkg -i "$OMADA_PKG"
|
||||||
rm -rf "${OMADA_PKG}"
|
rm -rf "$OMADA_PKG"
|
||||||
VERSION=$(sed -n 's/.*_v\([0-9.]*\)_.*_\([0-9]\{14\}\)\.deb$/\1-\2/p' <<<"${OMADA_PKG}")
|
|
||||||
echo "${VERSION}" >$HOME/.omada
|
|
||||||
msg_ok "Installed Omada Controller"
|
msg_ok "Installed Omada Controller"
|
||||||
|
|
||||||
motd_ssh
|
motd_ssh
|
||||||
|
|||||||
@@ -1,126 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
# Copyright (c) 2021-2026 community-scripts ORG
|
|
||||||
# Author: MickLesk (CanbiZ)
|
|
||||||
# License: MIT | https://github.com/community-scripts/ProxmoxVE/raw/main/LICENSE
|
|
||||||
# Source: https://shlink.io/
|
|
||||||
|
|
||||||
source /dev/stdin <<<"$FUNCTIONS_FILE_PATH"
|
|
||||||
color
|
|
||||||
verb_ip6
|
|
||||||
catch_errors
|
|
||||||
setting_up_container
|
|
||||||
network_check
|
|
||||||
update_os
|
|
||||||
|
|
||||||
PHP_VERSION="8.5" setup_php
|
|
||||||
setup_mariadb
|
|
||||||
MARIADB_DB_NAME="shlink" MARIADB_DB_USER="shlink" setup_mariadb_db
|
|
||||||
|
|
||||||
fetch_and_deploy_gh_release "shlink" "shlinkio/shlink" "prebuild" "latest" "/opt/shlink" "shlink*_php8.5_dist.zip"
|
|
||||||
|
|
||||||
msg_info "Setting up Application"
|
|
||||||
cd /opt/shlink
|
|
||||||
$STD php ./vendor/bin/rr get --no-interaction --location bin/
|
|
||||||
chmod +x bin/rr
|
|
||||||
mkdir -p data/cache data/locks data/log data/proxies data/temp-geolite
|
|
||||||
chmod -R 775 data
|
|
||||||
cat <<EOF >/opt/shlink/.env
|
|
||||||
DEFAULT_DOMAIN=${LOCAL_IP}:8080
|
|
||||||
IS_HTTPS_ENABLED=false
|
|
||||||
DB_DRIVER=maria
|
|
||||||
DB_NAME=${MARIADB_DB_NAME}
|
|
||||||
DB_USER=${MARIADB_DB_USER}
|
|
||||||
DB_PASSWORD=${MARIADB_DB_PASS}
|
|
||||||
DB_HOST=127.0.0.1
|
|
||||||
DB_PORT=3306
|
|
||||||
EOF
|
|
||||||
set -a
|
|
||||||
source /opt/shlink/.env
|
|
||||||
set +a
|
|
||||||
$STD php vendor/bin/shlink-installer init --no-interaction --clear-db-cache --skip-download-geolite
|
|
||||||
API_OUTPUT=$(php bin/cli api-key:generate --name=default 2>&1)
|
|
||||||
INITIAL_API_KEY=$(echo "$API_OUTPUT" | sed -n 's/.*Generated API key: "\([^"]*\)".*/\1/p')
|
|
||||||
if [[ -n "$INITIAL_API_KEY" ]]; then
|
|
||||||
echo "INITIAL_API_KEY=${INITIAL_API_KEY}" >>/opt/shlink/.env
|
|
||||||
fi
|
|
||||||
msg_ok "Set up Application"
|
|
||||||
|
|
||||||
if prompt_confirm "Install Shlink Web Client?" "y" 60; then
|
|
||||||
msg_info "Installing Dependencies"
|
|
||||||
$STD apt install -y nginx
|
|
||||||
msg_ok "Installed Dependencies"
|
|
||||||
|
|
||||||
fetch_and_deploy_gh_release "shlink-web-client" "shlinkio/shlink-web-client" "prebuild" "latest" "/opt/shlink-web-client" "shlink-web-client_*_dist.zip"
|
|
||||||
|
|
||||||
msg_info "Setting up Web Client"
|
|
||||||
cat <<EOF >/opt/shlink-web-client/servers.json
|
|
||||||
[
|
|
||||||
{
|
|
||||||
"name": "Shlink",
|
|
||||||
"url": "http://${LOCAL_IP}:8080",
|
|
||||||
"apiKey": "${INITIAL_API_KEY}"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
EOF
|
|
||||||
cat <<'EOF' >/etc/nginx/sites-available/shlink-web-client
|
|
||||||
server {
|
|
||||||
listen 3000 default_server;
|
|
||||||
charset utf-8;
|
|
||||||
root /opt/shlink-web-client;
|
|
||||||
index index.html;
|
|
||||||
|
|
||||||
location ~* \.(?:manifest|appcache|html?|xml|json)$ {
|
|
||||||
expires -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
location ~* \.(?:jpg|jpeg|gif|png|ico|cur|gz|svg|svgz|mp4|ogg|ogv|webm|htc)$ {
|
|
||||||
expires 1M;
|
|
||||||
add_header Cache-Control "public";
|
|
||||||
}
|
|
||||||
|
|
||||||
location ~* \.(?:css|js)$ {
|
|
||||||
expires 1y;
|
|
||||||
add_header Cache-Control "public";
|
|
||||||
}
|
|
||||||
|
|
||||||
location = /servers.json {
|
|
||||||
try_files /servers.json /conf.d/servers.json;
|
|
||||||
}
|
|
||||||
|
|
||||||
location / {
|
|
||||||
try_files $uri $uri/ /index.html$is_args$args;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
ln -sf /etc/nginx/sites-available/shlink-web-client /etc/nginx/sites-enabled/shlink-web-client
|
|
||||||
rm -f /etc/nginx/sites-enabled/default
|
|
||||||
systemctl enable -q nginx
|
|
||||||
$STD systemctl restart nginx
|
|
||||||
msg_ok "Set up Web Client"
|
|
||||||
fi
|
|
||||||
|
|
||||||
msg_info "Creating Service"
|
|
||||||
cat <<EOF >/etc/systemd/system/shlink.service
|
|
||||||
[Unit]
|
|
||||||
Description=Shlink URL Shortener
|
|
||||||
After=network.target mariadb.service
|
|
||||||
|
|
||||||
[Service]
|
|
||||||
Type=simple
|
|
||||||
User=root
|
|
||||||
WorkingDirectory=/opt/shlink
|
|
||||||
EnvironmentFile=/opt/shlink/.env
|
|
||||||
ExecStart=/opt/shlink/bin/rr serve -c config/roadrunner/.rr.yml
|
|
||||||
Restart=on-failure
|
|
||||||
RestartSec=5
|
|
||||||
|
|
||||||
[Install]
|
|
||||||
WantedBy=multi-user.target
|
|
||||||
EOF
|
|
||||||
systemctl enable -q --now shlink
|
|
||||||
msg_ok "Created Service"
|
|
||||||
|
|
||||||
motd_ssh
|
|
||||||
customize
|
|
||||||
cleanup_lxc
|
|
||||||
@@ -54,7 +54,7 @@ cd /opt
|
|||||||
rm -rf /opt/guacamole-server
|
rm -rf /opt/guacamole-server
|
||||||
msg_ok "Built Guacamole Server (guacd)"
|
msg_ok "Built Guacamole Server (guacd)"
|
||||||
|
|
||||||
NODE_VERSION="24" setup_nodejs
|
NODE_VERSION="22" setup_nodejs
|
||||||
fetch_and_deploy_gh_release "termix" "Termix-SSH/Termix" "tarball"
|
fetch_and_deploy_gh_release "termix" "Termix-SSH/Termix" "tarball"
|
||||||
|
|
||||||
msg_info "Building Frontend"
|
msg_info "Building Frontend"
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ msg_info "Installing Dependencies"
|
|||||||
$STD apt install -y build-essential
|
$STD apt install -y build-essential
|
||||||
msg_ok "Installed Dependencies"
|
msg_ok "Installed Dependencies"
|
||||||
|
|
||||||
NODE_VERSION="24" setup_nodejs
|
NODE_VERSION="22" setup_nodejs
|
||||||
fetch_and_deploy_gh_release "trek" "mauriceboe/TREK" "tarball"
|
fetch_and_deploy_gh_release "trek" "mauriceboe/TREK" "tarball"
|
||||||
|
|
||||||
msg_info "Building Client"
|
msg_info "Building Client"
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ setup_mariadb
|
|||||||
MARIADB_DB_NAME="wallabag" MARIADB_DB_USER="wallabag" setup_mariadb_db
|
MARIADB_DB_NAME="wallabag" MARIADB_DB_USER="wallabag" setup_mariadb_db
|
||||||
PHP_VERSION="8.3" PHP_FPM="YES" PHP_MODULE="tidy" setup_php
|
PHP_VERSION="8.3" PHP_FPM="YES" PHP_MODULE="tidy" setup_php
|
||||||
setup_composer
|
setup_composer
|
||||||
NODE_VERSION="24" setup_nodejs
|
NODE_VERSION="22" setup_nodejs
|
||||||
fetch_and_deploy_gh_release "wallabag" "wallabag/wallabag" "prebuild" "latest" "/opt/wallabag" "wallabag-*.tar.gz"
|
fetch_and_deploy_gh_release "wallabag" "wallabag/wallabag" "prebuild" "latest" "/opt/wallabag" "wallabag-*.tar.gz"
|
||||||
|
|
||||||
msg_info "Configuring Wallabag"
|
msg_info "Configuring Wallabag"
|
||||||
|
|||||||
@@ -18,10 +18,13 @@ export DEBIAN_FRONTEND=noninteractive
|
|||||||
$STD apt -y install --no-install-recommends \
|
$STD apt -y install --no-install-recommends \
|
||||||
git \
|
git \
|
||||||
traceroute \
|
traceroute \
|
||||||
build-essential \
|
make \
|
||||||
|
g++ \
|
||||||
|
traceroute \
|
||||||
xvfb \
|
xvfb \
|
||||||
dbus \
|
dbus \
|
||||||
xorg \
|
xorg \
|
||||||
|
xvfb \
|
||||||
gtk2-engines-pixbuf \
|
gtk2-engines-pixbuf \
|
||||||
dbus-x11 \
|
dbus-x11 \
|
||||||
xfonts-base \
|
xfonts-base \
|
||||||
@@ -40,13 +43,16 @@ rm -rf /usr/lib/python3.*/EXTERNALLY-MANAGED
|
|||||||
msg_ok "Setup Python3"
|
msg_ok "Setup Python3"
|
||||||
|
|
||||||
msg_info "Installing Chromium"
|
msg_info "Installing Chromium"
|
||||||
setup_deb822_repo \
|
curl -fsSL https://dl-ssl.google.com/linux/linux_signing_key.pub | gpg --dearmor -o /usr/share/keyrings/google-chrome-keyring.gpg
|
||||||
"google-chrome" \
|
cat <<EOF | sudo tee /etc/apt/sources.list.d/google-chrome.sources >/dev/null
|
||||||
"https://dl-ssl.google.com/linux/linux_signing_key.pub" \
|
Types: deb
|
||||||
"http://dl.google.com/linux/chrome/deb/" \
|
URIs: http://dl.google.com/linux/chrome/deb/
|
||||||
"stable" \
|
Suites: stable
|
||||||
"main" \
|
Components: main
|
||||||
"amd64"
|
Architectures: amd64
|
||||||
|
Signed-By: /usr/share/keyrings/google-chrome-keyring.gpg
|
||||||
|
EOF
|
||||||
|
$STD apt update
|
||||||
$STD apt -y install \
|
$STD apt -y install \
|
||||||
chromium \
|
chromium \
|
||||||
libxss1 \
|
libxss1 \
|
||||||
@@ -58,14 +64,13 @@ msg_info "Setting up Chromium"
|
|||||||
chmod 755 /usr/bin/chromium
|
chmod 755 /usr/bin/chromium
|
||||||
msg_ok "Setup Chromium"
|
msg_ok "Setup Chromium"
|
||||||
|
|
||||||
fetch_and_deploy_gh_release "web-check" "Lissy93/web-check" "tarball"
|
fetch_and_deploy_gh_release "web-check" "CrazyWolf13/web-check" "tarball"
|
||||||
|
|
||||||
msg_info "Installing Web-Check (Patience)"
|
msg_info "Installing Web-Check (Patience)"
|
||||||
cd /opt/web-check
|
cd /opt/web-check
|
||||||
cat <<'EOF' >/opt/web-check/.env
|
cat <<'EOF' >/opt/web-check/.env
|
||||||
CHROME_PATH=/usr/bin/chromium
|
CHROME_PATH=/usr/bin/chromium
|
||||||
PUPPETEER_EXECUTABLE_PATH=/usr/bin/chromium
|
PUPPETEER_EXECUTABLE_PATH=/usr/bin/chromium
|
||||||
PUPPETEER_SKIP_DOWNLOAD='true'
|
|
||||||
HEADLESS=true
|
HEADLESS=true
|
||||||
GOOGLE_CLOUD_API_KEY=''
|
GOOGLE_CLOUD_API_KEY=''
|
||||||
REACT_APP_SHODAN_API_KEY=''
|
REACT_APP_SHODAN_API_KEY=''
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ msg_info "Installing Dependencies"
|
|||||||
$STD apt install -y git
|
$STD apt install -y git
|
||||||
msg_ok "Installed Dependencies"
|
msg_ok "Installed Dependencies"
|
||||||
|
|
||||||
NODE_VERSION="24" NODE_MODULE="yarn,node-gyp" setup_nodejs
|
NODE_VERSION="22" NODE_MODULE="yarn,node-gyp" setup_nodejs
|
||||||
PG_VERSION="17" setup_postgresql
|
PG_VERSION="17" setup_postgresql
|
||||||
PG_DB_NAME="wiki" PG_DB_USER="wikijs_user" PG_DB_EXTENSIONS="pg_trgm" setup_postgresql_db
|
PG_DB_NAME="wiki" PG_DB_USER="wikijs_user" PG_DB_EXTENSIONS="pg_trgm" setup_postgresql_db
|
||||||
fetch_and_deploy_gh_release "wikijs" "requarks/wiki" "prebuild" "latest" "/opt/wikijs" "wiki-js.tar.gz"
|
fetch_and_deploy_gh_release "wikijs" "requarks/wiki" "prebuild" "latest" "/opt/wikijs" "wiki-js.tar.gz"
|
||||||
|
|||||||
109
misc/build.func
109
misc/build.func
@@ -1062,7 +1062,6 @@ load_vars_file() {
|
|||||||
var_gateway var_hostname var_ipv6_method var_mac var_mknod var_mount_fs var_mtu
|
var_gateway var_hostname var_ipv6_method var_mac var_mknod var_mount_fs var_mtu
|
||||||
var_net var_nesting var_ns var_os var_protection var_pw var_ram var_tags var_timezone var_tun var_unprivileged
|
var_net var_nesting var_ns var_os var_protection var_pw var_ram var_tags var_timezone var_tun var_unprivileged
|
||||||
var_verbose var_version var_vlan var_ssh var_ssh_authorized_key var_container_storage var_template_storage var_searchdomain
|
var_verbose var_version var_vlan var_ssh var_ssh_authorized_key var_container_storage var_template_storage var_searchdomain
|
||||||
var_post_install
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Whitelist check helper
|
# Whitelist check helper
|
||||||
@@ -1280,7 +1279,6 @@ default_var_settings() {
|
|||||||
var_gateway var_hostname var_ipv6_method var_mac var_mknod var_mount_fs var_mtu
|
var_gateway var_hostname var_ipv6_method var_mac var_mknod var_mount_fs var_mtu
|
||||||
var_net var_nesting var_ns var_os var_protection var_pw var_ram var_tags var_timezone var_tun var_unprivileged
|
var_net var_nesting var_ns var_os var_protection var_pw var_ram var_tags var_timezone var_tun var_unprivileged
|
||||||
var_verbose var_version var_vlan var_ssh var_ssh_authorized_key var_container_storage var_template_storage
|
var_verbose var_version var_vlan var_ssh var_ssh_authorized_key var_container_storage var_template_storage
|
||||||
var_post_install
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Snapshot: environment variables (highest precedence)
|
# Snapshot: environment variables (highest precedence)
|
||||||
@@ -1376,11 +1374,6 @@ var_verbose=no
|
|||||||
# GitHub Personal Access Token (optional – avoids API rate limits during installs)
|
# GitHub Personal Access Token (optional – avoids API rate limits during installs)
|
||||||
# Create at https://github.com/settings/tokens – read-only public access is sufficient
|
# Create at https://github.com/settings/tokens – read-only public access is sufficient
|
||||||
# var_github_token=ghp_your_token_here
|
# var_github_token=ghp_your_token_here
|
||||||
|
|
||||||
# Optional post-install script (host-side path to a *.sh on the Proxmox host)
|
|
||||||
# Runs ON THE HOST after the container is fully provisioned.
|
|
||||||
# Available env vars: APP, NSAPP, CTID, IP, HN, STORAGE, BRG
|
|
||||||
# var_post_install=/opt/post-install/myhook.sh
|
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
# Now choose storages (always prompt unless just one exists)
|
# Now choose storages (always prompt unless just one exists)
|
||||||
@@ -1459,7 +1452,6 @@ if ! declare -p VAR_WHITELIST >/dev/null 2>&1; then
|
|||||||
var_gateway var_hostname var_ipv6_method var_mac var_mknod var_mount_fs var_mtu
|
var_gateway var_hostname var_ipv6_method var_mac var_mknod var_mount_fs var_mtu
|
||||||
var_net var_nesting var_ns var_os var_protection var_pw var_ram var_tags var_timezone var_tun var_unprivileged
|
var_net var_nesting var_ns var_os var_protection var_pw var_ram var_tags var_timezone var_tun var_unprivileged
|
||||||
var_verbose var_version var_vlan var_ssh var_ssh_authorized_key var_container_storage var_template_storage var_searchdomain
|
var_verbose var_version var_vlan var_ssh var_ssh_authorized_key var_container_storage var_template_storage var_searchdomain
|
||||||
var_post_install
|
|
||||||
)
|
)
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -1672,7 +1664,6 @@ _build_current_app_vars_tmp() {
|
|||||||
|
|
||||||
[ -n "$_tpl_storage" ] && echo "var_template_storage=$(_sanitize_value "$_tpl_storage")"
|
[ -n "$_tpl_storage" ] && echo "var_template_storage=$(_sanitize_value "$_tpl_storage")"
|
||||||
[ -n "$_ct_storage" ] && echo "var_container_storage=$(_sanitize_value "$_ct_storage")"
|
[ -n "$_ct_storage" ] && echo "var_container_storage=$(_sanitize_value "$_ct_storage")"
|
||||||
[ -n "${var_post_install:-}" ] && echo "var_post_install=$(_sanitize_value "${var_post_install}")"
|
|
||||||
} >"$tmpf"
|
} >"$tmpf"
|
||||||
|
|
||||||
echo "$tmpf"
|
echo "$tmpf"
|
||||||
@@ -1817,7 +1808,7 @@ advanced_settings() {
|
|||||||
TAGS="community-script${var_tags:+;${var_tags}}"
|
TAGS="community-script${var_tags:+;${var_tags}}"
|
||||||
fi
|
fi
|
||||||
local STEP=1
|
local STEP=1
|
||||||
local MAX_STEP=29
|
local MAX_STEP=28
|
||||||
|
|
||||||
# Store values for back navigation - inherit from var_* app defaults
|
# Store values for back navigation - inherit from var_* app defaults
|
||||||
local _ct_type="${var_unprivileged:-1}"
|
local _ct_type="${var_unprivileged:-1}"
|
||||||
@@ -1851,7 +1842,6 @@ advanced_settings() {
|
|||||||
local _enable_mknod="${var_mknod:-0}"
|
local _enable_mknod="${var_mknod:-0}"
|
||||||
local _mount_fs="${var_mount_fs:-}"
|
local _mount_fs="${var_mount_fs:-}"
|
||||||
local _protect_ct="${var_protection:-no}"
|
local _protect_ct="${var_protection:-no}"
|
||||||
local _post_install="${var_post_install:-}"
|
|
||||||
|
|
||||||
# Detect host timezone for default (if not set via var_timezone)
|
# Detect host timezone for default (if not set via var_timezone)
|
||||||
local _host_timezone=""
|
local _host_timezone=""
|
||||||
@@ -2709,61 +2699,9 @@ advanced_settings() {
|
|||||||
;;
|
;;
|
||||||
|
|
||||||
# ═══════════════════════════════════════════════════════════════════════════
|
# ═══════════════════════════════════════════════════════════════════════════
|
||||||
# STEP 28: Optional host-side post-install hook (path on the Proxmox HOST)
|
# STEP 28: Verbose Mode & Confirmation
|
||||||
# ═══════════════════════════════════════════════════════════════════════════
|
# ═══════════════════════════════════════════════════════════════════════════
|
||||||
28)
|
28)
|
||||||
local _hook_prompt="Optional: absolute path to a *.sh file ON THE PROXMOX HOST.
|
|
||||||
|
|
||||||
It runs as root on the HOST (NOT in the LXC) after the container
|
|
||||||
is fully provisioned and started.
|
|
||||||
|
|
||||||
Available env vars: APP, NSAPP, CTID, IP, HN, STORAGE, BRG.
|
|
||||||
|
|
||||||
Leave empty to skip."
|
|
||||||
while true; do
|
|
||||||
if result=$(whiptail --backtitle "Proxmox VE Helper Scripts [Step $STEP/$MAX_STEP]" \
|
|
||||||
--title "POST-INSTALL HOOK (HOST)" \
|
|
||||||
--ok-button "Next" --cancel-button "Back" \
|
|
||||||
--inputbox "$_hook_prompt" 16 70 "${_post_install}" \
|
|
||||||
3>&1 1>&2 2>&3); then
|
|
||||||
# Normalize: strip surrounding whitespace
|
|
||||||
result="$(printf '%s' "$result" | sed -e 's/^[[:space:]]*//' -e 's/[[:space:]]*$//')"
|
|
||||||
if [[ -z "$result" ]]; then
|
|
||||||
_post_install=""
|
|
||||||
((STEP++))
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
# Reject obvious shell-meta sneaking through
|
|
||||||
if [[ "$result" == *';'* || "$result" == *'$('* || "$result" == *'`'* || "$result" == *'&&'* || "$result" == *'||'* ]]; then
|
|
||||||
whiptail --backtitle "Proxmox VE Helper Scripts" --title "INVALID PATH" \
|
|
||||||
--msgbox "Path contains shell metacharacters. Please provide a plain absolute file path." 10 70
|
|
||||||
continue
|
|
||||||
fi
|
|
||||||
if [[ "$result" != /* ]]; then
|
|
||||||
whiptail --backtitle "Proxmox VE Helper Scripts" --title "INVALID PATH" \
|
|
||||||
--msgbox "Path must be absolute (start with /).\n\nGot: $result" 10 70
|
|
||||||
continue
|
|
||||||
fi
|
|
||||||
if [[ ! -f "$result" ]]; then
|
|
||||||
if ! whiptail --backtitle "Proxmox VE Helper Scripts" --title "FILE NOT FOUND" \
|
|
||||||
--yesno "File does not exist on host:\n\n$result\n\nKeep this path anyway?" 12 70; then
|
|
||||||
continue
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
_post_install="$result"
|
|
||||||
((STEP++))
|
|
||||||
break
|
|
||||||
else
|
|
||||||
((STEP--))
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
;;
|
|
||||||
|
|
||||||
# ═══════════════════════════════════════════════════════════════════════════
|
|
||||||
# STEP 29: Verbose Mode & Confirmation
|
|
||||||
# ═══════════════════════════════════════════════════════════════════════════
|
|
||||||
29)
|
|
||||||
local verbose_default_flag="--defaultno"
|
local verbose_default_flag="--defaultno"
|
||||||
[[ "$_verbose" == "yes" ]] && verbose_default_flag=""
|
[[ "$_verbose" == "yes" ]] && verbose_default_flag=""
|
||||||
|
|
||||||
@@ -2792,11 +2730,6 @@ Leave empty to skip."
|
|||||||
local apt_display="${_apt_cacher:-no}"
|
local apt_display="${_apt_cacher:-no}"
|
||||||
[[ "$_apt_cacher" == "yes" && -n "$_apt_cacher_ip" ]] && apt_display="$_apt_cacher_ip"
|
[[ "$_apt_cacher" == "yes" && -n "$_apt_cacher_ip" ]] && apt_display="$_apt_cacher_ip"
|
||||||
|
|
||||||
local post_install_display="${_post_install:-(none)}"
|
|
||||||
local post_install_warn=""
|
|
||||||
[[ -n "$_post_install" ]] && post_install_warn="
|
|
||||||
⚠ Hook runs as root on Proxmox HOST (not in LXC)"
|
|
||||||
|
|
||||||
local summary="Container Type: $ct_type_desc
|
local summary="Container Type: $ct_type_desc
|
||||||
Container ID: $_ct_id
|
Container ID: $_ct_id
|
||||||
Hostname: $_hostname
|
Hostname: $_hostname
|
||||||
@@ -2820,8 +2753,7 @@ Features:
|
|||||||
Advanced:
|
Advanced:
|
||||||
Timezone: $tz_display
|
Timezone: $tz_display
|
||||||
APT Cacher: $apt_display
|
APT Cacher: $apt_display
|
||||||
Verbose: $_verbose
|
Verbose: $_verbose"
|
||||||
Post-Install Script: ${post_install_display}${post_install_warn}"
|
|
||||||
|
|
||||||
if whiptail --backtitle "Proxmox VE Helper Scripts [Step $STEP/$MAX_STEP]" \
|
if whiptail --backtitle "Proxmox VE Helper Scripts [Step $STEP/$MAX_STEP]" \
|
||||||
--title "CONFIRM SETTINGS" \
|
--title "CONFIRM SETTINGS" \
|
||||||
@@ -2864,7 +2796,6 @@ Advanced:
|
|||||||
APT_CACHER="$_apt_cacher"
|
APT_CACHER="$_apt_cacher"
|
||||||
APT_CACHER_IP="$_apt_cacher_ip"
|
APT_CACHER_IP="$_apt_cacher_ip"
|
||||||
VERBOSE="$_verbose"
|
VERBOSE="$_verbose"
|
||||||
var_post_install="$_post_install"
|
|
||||||
|
|
||||||
# Update var_* based on user choice (for functions that check these)
|
# Update var_* based on user choice (for functions that check these)
|
||||||
var_gpu="$_enable_gpu"
|
var_gpu="$_enable_gpu"
|
||||||
@@ -6430,40 +6361,6 @@ EOF
|
|||||||
systemctl start ping-instances.service
|
systemctl start ping-instances.service
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Optional host-side post-install hook
|
|
||||||
# Path comes from var_post_install (default.vars / app.vars / advanced settings).
|
|
||||||
# Runs ON THE PROXMOX HOST after the container is up and configured.
|
|
||||||
# Exposed env vars: APP, NSAPP, CTID, IP, HN, STORAGE, BRG.
|
|
||||||
# Output (stdout/stderr) is captured to /var/log/community-scripts/post-install-<CTID>.log
|
|
||||||
if [[ -n "${var_post_install:-}" ]]; then
|
|
||||||
local _hook_log_dir="/var/log/community-scripts"
|
|
||||||
local _hook_log="${_hook_log_dir}/post-install-${CTID}.log"
|
|
||||||
mkdir -p "$_hook_log_dir" 2>/dev/null || true
|
|
||||||
|
|
||||||
if [[ ! -f "${var_post_install}" ]]; then
|
|
||||||
msg_error "Post-install hook not found on host: ${var_post_install}"
|
|
||||||
whiptail --backtitle "Proxmox VE Helper Scripts" \
|
|
||||||
--title "POST-INSTALL HOOK FAILED" \
|
|
||||||
--msgbox "The configured post-install hook was not found on the Proxmox host:\n\n${var_post_install}\n\nThe LXC was created successfully, but the hook did NOT run." 14 72 || true
|
|
||||||
else
|
|
||||||
msg_info "Running post-install hook: ${var_post_install}"
|
|
||||||
local _hook_rc=0
|
|
||||||
APP="$APP" NSAPP="${NSAPP:-}" CTID="$CTID" IP="$IP" HN="${HN:-}" \
|
|
||||||
STORAGE="${STORAGE:-}" BRG="${BRG:-}" \
|
|
||||||
bash "${var_post_install}" >"${_hook_log}" 2>&1 || _hook_rc=$?
|
|
||||||
if [[ $_hook_rc -eq 0 ]]; then
|
|
||||||
msg_ok "Post-install hook completed (log: ${_hook_log})"
|
|
||||||
else
|
|
||||||
msg_error "Post-install hook failed (rc=${_hook_rc}) – see ${_hook_log}"
|
|
||||||
local _hook_tail=""
|
|
||||||
_hook_tail="$(tail -n 15 "${_hook_log}" 2>/dev/null || true)"
|
|
||||||
whiptail --backtitle "Proxmox VE Helper Scripts" \
|
|
||||||
--title "POST-INSTALL HOOK FAILED" \
|
|
||||||
--msgbox "Hook exited with code ${_hook_rc}.\n\nScript: ${var_post_install}\nLog: ${_hook_log}\n\n--- Last log lines ---\n${_hook_tail}\n\nThe LXC itself was created successfully." 22 78 || true
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
INSTALL_COMPLETE=true
|
INSTALL_COMPLETE=true
|
||||||
post_update_to_api "done" "none"
|
post_update_to_api "done" "none"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -147,7 +147,7 @@ EOF
|
|||||||
# Create update script
|
# Create update script
|
||||||
msg_info "Creating update script"
|
msg_info "Creating update script"
|
||||||
ensure_usr_local_bin_persist
|
ensure_usr_local_bin_persist
|
||||||
cat <<'EOF' >/usr/local/bin/update_cronmaster
|
cat <<EOF >/usr/local/bin/update_cronmaster
|
||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
# CronMaster Update Script
|
# CronMaster Update Script
|
||||||
type=update bash -c "$(curl -fsSL https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/tools/addon/cronmaster.sh)"
|
type=update bash -c "$(curl -fsSL https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/tools/addon/cronmaster.sh)"
|
||||||
|
|||||||
@@ -68,24 +68,6 @@ function uninstall() {
|
|||||||
# ==============================================================================
|
# ==============================================================================
|
||||||
function update() {
|
function update() {
|
||||||
if check_for_gh_release "qbittorrent-exporter" "martabal/qbittorrent-exporter"; then
|
if check_for_gh_release "qbittorrent-exporter" "martabal/qbittorrent-exporter"; then
|
||||||
if [[ "$(printf '%s\n' "2.0.0" "$CHECK_UPDATE_RELEASE" | sort -V | tail -n1)" == "$CHECK_UPDATE_RELEASE" ]] && \
|
|
||||||
! grep -q "QBITTORRENT_API_KEY" "$CONFIG_PATH" 2>/dev/null; then
|
|
||||||
echo ""
|
|
||||||
msg_warn "Version 2.0.0 introduces a breaking change: username/password login has been replaced by an API key."
|
|
||||||
echo -e "${TAB3}${INFO} You must create an API key in qBittorrent under Tools > Options > Web UI > API key"
|
|
||||||
echo ""
|
|
||||||
echo -n "${TAB3}Enter your qBittorrent API key (or press Enter to abort): "
|
|
||||||
read -r QBITTORRENT_API_KEY
|
|
||||||
if [[ -z "$QBITTORRENT_API_KEY" ]]; then
|
|
||||||
msg_warn "No API key provided. Update aborted."
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
sed -i '/^QBITTORRENT_USERNAME=/d' "$CONFIG_PATH"
|
|
||||||
sed -i '/^QBITTORRENT_PASSWORD=/d' "$CONFIG_PATH"
|
|
||||||
echo "QBITTORRENT_API_KEY=\"${QBITTORRENT_API_KEY}\"" >>"$CONFIG_PATH"
|
|
||||||
msg_ok "API key saved to configuration"
|
|
||||||
fi
|
|
||||||
|
|
||||||
msg_info "Stopping service"
|
msg_info "Stopping service"
|
||||||
if [[ "$OS" == "Alpine" ]]; then
|
if [[ "$OS" == "Alpine" ]]; then
|
||||||
rc-service qbittorrent-exporter stop &>/dev/null
|
rc-service qbittorrent-exporter stop &>/dev/null
|
||||||
@@ -118,9 +100,10 @@ function update() {
|
|||||||
# INSTALL
|
# INSTALL
|
||||||
# ==============================================================================
|
# ==============================================================================
|
||||||
function install() {
|
function install() {
|
||||||
read -erp "${TAB3}Enter URL of qBittorrent, example: (http://127.0.0.1:8080): " QBITTORRENT_BASE_URL
|
read -erp "Enter URL of qBittorrent, example: (http://127.0.0.1:8080): " QBITTORRENT_BASE_URL
|
||||||
echo -e "${TAB3}${INFO} Create an API key in qBittorrent under Tools > Options > Web UI > API key"
|
read -erp "Enter qBittorrent username: " QBITTORRENT_USERNAME
|
||||||
read -erp "${TAB3}Enter qBittorrent API key: " QBITTORRENT_API_KEY
|
read -rsp "Enter qBittorrent password: " QBITTORRENT_PASSWORD
|
||||||
|
printf "\n"
|
||||||
|
|
||||||
fetch_and_deploy_gh_release "qbittorrent-exporter" "martabal/qbittorrent-exporter" "tarball" "latest"
|
fetch_and_deploy_gh_release "qbittorrent-exporter" "martabal/qbittorrent-exporter" "tarball" "latest"
|
||||||
setup_go
|
setup_go
|
||||||
@@ -133,7 +116,8 @@ function install() {
|
|||||||
cat <<EOF >"$CONFIG_PATH"
|
cat <<EOF >"$CONFIG_PATH"
|
||||||
# https://github.com/martabal/qbittorrent-exporter?tab=readme-ov-file#parameters
|
# https://github.com/martabal/qbittorrent-exporter?tab=readme-ov-file#parameters
|
||||||
QBITTORRENT_BASE_URL="${QBITTORRENT_BASE_URL}"
|
QBITTORRENT_BASE_URL="${QBITTORRENT_BASE_URL}"
|
||||||
QBITTORRENT_API_KEY="${QBITTORRENT_API_KEY}"
|
QBITTORRENT_USERNAME="${QBITTORRENT_USERNAME}"
|
||||||
|
QBITTORRENT_PASSWORD="${QBITTORRENT_PASSWORD}"
|
||||||
EOF
|
EOF
|
||||||
msg_ok "Created configuration"
|
msg_ok "Created configuration"
|
||||||
|
|
||||||
|
|||||||
@@ -1,436 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
# ============================================================================
|
|
||||||
# Community-Scripts ProxmoxVE — Post-Install Hook: Example Library
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
# This file is NOT meant to be executed as-is.
|
|
||||||
# It is a collection of complete, copy-pasteable example hooks for the
|
|
||||||
# optional `var_post_install` feature in build.func.
|
|
||||||
#
|
|
||||||
# HOW IT WORKS
|
|
||||||
# ------------
|
|
||||||
# In the ct/*.sh CT scripts (or via Advanced Settings → Step 28) you can
|
|
||||||
# point `var_post_install` to an absolute path on the Proxmox HOST, e.g.:
|
|
||||||
#
|
|
||||||
# # in /root/.community-scripts/default.vars
|
|
||||||
# var_post_install=/opt/community-scripts/hooks/notify.sh
|
|
||||||
#
|
|
||||||
# # OR per-app, in app.vars
|
|
||||||
# var_post_install=/opt/community-scripts/hooks/vaultwarden-postprovision.sh
|
|
||||||
#
|
|
||||||
# # OR interactively in the Advanced Settings whiptail (Step 28).
|
|
||||||
#
|
|
||||||
# The hook runs ON THE PROXMOX HOST (NOT inside the LXC) as root,
|
|
||||||
# AFTER the container is fully provisioned, started and the description
|
|
||||||
# is set. stdout/stderr is captured to:
|
|
||||||
#
|
|
||||||
# /var/log/community-scripts/post-install-<CTID>.log
|
|
||||||
#
|
|
||||||
# AVAILABLE ENV VARIABLES
|
|
||||||
# -----------------------
|
|
||||||
# APP - Pretty name (e.g. "Vaultwarden")
|
|
||||||
# NSAPP - Slug / lowercase (e.g. "vaultwarden")
|
|
||||||
# CTID - Numeric container ID (e.g. "103")
|
|
||||||
# IP - IPv4 address of the LXC (e.g. "192.168.1.50")
|
|
||||||
# HN - Hostname (e.g. "vaultwarden")
|
|
||||||
# STORAGE - Storage where the rootfs lives (e.g. "local-lvm")
|
|
||||||
# BRG - Bridge (e.g. "vmbr0")
|
|
||||||
#
|
|
||||||
# GENERAL TIPS
|
|
||||||
# ------------
|
|
||||||
# - Use `set -euo pipefail` so failures actually surface.
|
|
||||||
# - Use `|| true` on best-effort steps you do not want to abort the hook.
|
|
||||||
# - The file just needs to be a valid script. `+x` is optional — it is
|
|
||||||
# invoked via `bash <path>`. Shebang is honored only if you call it
|
|
||||||
# yourself; otherwise the shebang line is purely cosmetic.
|
|
||||||
# - If the hook exits non-zero, the user gets a whiptail popup with the
|
|
||||||
# last 15 log lines. The LXC creation itself is NOT rolled back.
|
|
||||||
# - Keep hooks idempotent — they may be re-run if you recreate a CT.
|
|
||||||
#
|
|
||||||
# HOW TO USE THIS FILE
|
|
||||||
# --------------------
|
|
||||||
# 1. Copy ONE example block (between the BEGIN/END markers) into a new
|
|
||||||
# file on the Proxmox host, e.g. /opt/community-scripts/hooks/notify.sh
|
|
||||||
# 2. chmod +x /opt/community-scripts/hooks/notify.sh (optional)
|
|
||||||
# 3. Set var_post_install in default.vars / app.vars or pick the path
|
|
||||||
# in Advanced Settings.
|
|
||||||
# ============================================================================
|
|
||||||
|
|
||||||
# ============================================================================
|
|
||||||
# ▼▼▼ EXAMPLE 1 — BEGIN ▼▼▼
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
# Name : minimal-logger.sh
|
|
||||||
# Purpose : Append every newly created LXC to a single CSV-ish log.
|
|
||||||
# Difficulty : ⭐ Beginner
|
|
||||||
# Side effects: Writes to /var/log/community-scripts/created-lxcs.log
|
|
||||||
# Use case : You just want a paper trail of "what got created when".
|
|
||||||
# ============================================================================
|
|
||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
LOG_DIR="/var/log/community-scripts"
|
|
||||||
LOG_FILE="${LOG_DIR}/created-lxcs.log"
|
|
||||||
|
|
||||||
mkdir -p "$LOG_DIR"
|
|
||||||
|
|
||||||
# Header on first use
|
|
||||||
if [[ ! -s "$LOG_FILE" ]]; then
|
|
||||||
echo "timestamp;ctid;app;hostname;ip;bridge;storage" >"$LOG_FILE"
|
|
||||||
fi
|
|
||||||
|
|
||||||
printf '%s;%s;%s;%s;%s;%s;%s\n' \
|
|
||||||
"$(date -Iseconds)" \
|
|
||||||
"${CTID}" \
|
|
||||||
"${APP}" \
|
|
||||||
"${HN}" \
|
|
||||||
"${IP}" \
|
|
||||||
"${BRG}" \
|
|
||||||
"${STORAGE}" \
|
|
||||||
>>"$LOG_FILE"
|
|
||||||
|
|
||||||
echo "Logged ${APP} (CTID=${CTID}) to ${LOG_FILE}"
|
|
||||||
# ▲▲▲ EXAMPLE 1 — END ▲▲▲
|
|
||||||
|
|
||||||
# ============================================================================
|
|
||||||
# ▼▼▼ EXAMPLE 2 — BEGIN ▼▼▼
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
# Name : discord-gotify-notify.sh
|
|
||||||
# Purpose : Send a rich Discord embed AND a Gotify push notification
|
|
||||||
# whenever a new LXC is provisioned.
|
|
||||||
# Difficulty : ⭐⭐ Intermediate
|
|
||||||
# Requires : curl on the host (default), reachable webhook URLs.
|
|
||||||
# Side effects: Outbound HTTPS to Discord + your Gotify server.
|
|
||||||
# ============================================================================
|
|
||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
# --- CONFIG (edit me) -------------------------------------------------------
|
|
||||||
DISCORD_WEBHOOK="https://discord.com/api/webhooks/XXXXXXXX/YYYYYYYY"
|
|
||||||
GOTIFY_URL="https://gotify.example.com"
|
|
||||||
GOTIFY_TOKEN="AbCdEfGhIjKlMnO"
|
|
||||||
GOTIFY_PRIORITY=5
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
# Resolve the Proxmox node's hostname for context
|
|
||||||
NODE="$(hostname -s)"
|
|
||||||
TS="$(date -Iseconds)"
|
|
||||||
|
|
||||||
# --- Discord embed ----------------------------------------------------------
|
|
||||||
read -r -d '' DISCORD_PAYLOAD <<JSON || true
|
|
||||||
{
|
|
||||||
"username": "Proxmox - ${NODE}",
|
|
||||||
"avatar_url": "https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/misc/images/logo-81x112.png",
|
|
||||||
"embeds": [{
|
|
||||||
"title": "✅ ${APP} LXC created",
|
|
||||||
"description": "A new community-script LXC has been provisioned on **${NODE}**.",
|
|
||||||
"color": 3066993,
|
|
||||||
"timestamp": "${TS}",
|
|
||||||
"fields": [
|
|
||||||
{"name": "CTID", "value": "${CTID}", "inline": true},
|
|
||||||
{"name": "Hostname", "value": "${HN}", "inline": true},
|
|
||||||
{"name": "App", "value": "${APP}", "inline": true},
|
|
||||||
{"name": "IP", "value": "${IP}", "inline": true},
|
|
||||||
{"name": "Bridge", "value": "${BRG}", "inline": true},
|
|
||||||
{"name": "Storage", "value": "${STORAGE}", "inline": true}
|
|
||||||
],
|
|
||||||
"footer": {"text": "community-scripts.org"}
|
|
||||||
}]
|
|
||||||
}
|
|
||||||
JSON
|
|
||||||
|
|
||||||
curl -fsS --max-time 10 \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
-X POST "$DISCORD_WEBHOOK" \
|
|
||||||
--data "$DISCORD_PAYLOAD" \
|
|
||||||
>/dev/null ||
|
|
||||||
echo "WARN: Discord webhook failed (non-fatal)"
|
|
||||||
|
|
||||||
# --- Gotify push ------------------------------------------------------------
|
|
||||||
curl -fsS --max-time 10 \
|
|
||||||
-H "X-Gotify-Key: ${GOTIFY_TOKEN}" \
|
|
||||||
-F "title=Proxmox: ${APP} LXC created" \
|
|
||||||
-F "message=CTID=${CTID} IP=${IP} HN=${HN} on ${NODE}" \
|
|
||||||
-F "priority=${GOTIFY_PRIORITY}" \
|
|
||||||
"${GOTIFY_URL}/message" \
|
|
||||||
>/dev/null ||
|
|
||||||
echo "WARN: Gotify push failed (non-fatal)"
|
|
||||||
|
|
||||||
echo "Notifications dispatched for CTID=${CTID}"
|
|
||||||
# ▲▲▲ EXAMPLE 2 — END ▲▲▲
|
|
||||||
|
|
||||||
# ============================================================================
|
|
||||||
# ▼▼▼ EXAMPLE 3 — BEGIN ▼▼▼
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
# Name : auto-pool-tags-backup.sh
|
|
||||||
# Purpose : Add the new LXC to a Proxmox pool, append cluster-wide tags,
|
|
||||||
# register a DNS record in pi-hole, and trigger an immediate
|
|
||||||
# snapshot backup to a configured storage.
|
|
||||||
# Difficulty : ⭐⭐⭐ Advanced
|
|
||||||
# Requires : pvesh, pct, vzdump (host-side; available by default on PVE),
|
|
||||||
# a reachable pi-hole admin API.
|
|
||||||
# ============================================================================
|
|
||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
# --- CONFIG (edit me) -------------------------------------------------------
|
|
||||||
TARGET_POOL="auto-lxc"
|
|
||||||
EXTRA_TAGS=("auto-provisioned" "${NSAPP}") # community-script tag is set by build.func
|
|
||||||
BACKUP_STORAGE="pbs-main" # set to "" to skip initial backup
|
|
||||||
PIHOLE_HOST="192.168.1.5"
|
|
||||||
PIHOLE_PASSWORD="changeme" # web-UI password
|
|
||||||
DNS_DOMAIN="lan" # FQDN will be ${HN}.${DNS_DOMAIN}
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
# 1) Ensure the pool exists, then attach the CT
|
|
||||||
if ! pvesh get "/pools/${TARGET_POOL}" >/dev/null 2>&1; then
|
|
||||||
echo "Creating pool: ${TARGET_POOL}"
|
|
||||||
pvesh create /pools --poolid "${TARGET_POOL}" --comment "Auto-created by post-install hook" || true
|
|
||||||
fi
|
|
||||||
echo "Adding CTID=${CTID} to pool=${TARGET_POOL}"
|
|
||||||
pvesh set "/pools/${TARGET_POOL}" --vms "${CTID}" || echo "WARN: pool attach failed (non-fatal)"
|
|
||||||
|
|
||||||
# 2) Merge new tags with existing ones (preserve community-script etc.)
|
|
||||||
CURRENT_TAGS="$(pct config "${CTID}" | awk -F': ' '/^tags:/{print $2}')"
|
|
||||||
declare -A TAG_SET
|
|
||||||
IFS=';' read -r -a CUR_ARR <<<"${CURRENT_TAGS:-}"
|
|
||||||
for t in "${CUR_ARR[@]}"; do [[ -n "$t" ]] && TAG_SET["$t"]=1; done
|
|
||||||
for t in "${EXTRA_TAGS[@]}"; do [[ -n "$t" ]] && TAG_SET["$t"]=1; done
|
|
||||||
NEW_TAGS="$(
|
|
||||||
IFS=';'
|
|
||||||
echo "${!TAG_SET[*]}"
|
|
||||||
)"
|
|
||||||
echo "Setting tags: ${NEW_TAGS}"
|
|
||||||
pct set "${CTID}" --tags "${NEW_TAGS}" || echo "WARN: tag update failed (non-fatal)"
|
|
||||||
|
|
||||||
# 3) Register DNS in pi-hole (custom DNS record)
|
|
||||||
FQDN="${HN}.${DNS_DOMAIN}"
|
|
||||||
echo "Registering DNS: ${FQDN} → ${IP} on pi-hole ${PIHOLE_HOST}"
|
|
||||||
SID="$(curl -fsS --max-time 5 \
|
|
||||||
-d "pw=${PIHOLE_PASSWORD}" \
|
|
||||||
"http://${PIHOLE_HOST}/api/auth" 2>/dev/null |
|
|
||||||
sed -nE 's/.*"sid":"([^"]+)".*/\1/p' || true)"
|
|
||||||
|
|
||||||
if [[ -n "${SID}" ]]; then
|
|
||||||
curl -fsS --max-time 5 -X PUT \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
-H "sid: ${SID}" \
|
|
||||||
-d "{\"hosts\":[\"${IP} ${FQDN}\"]}" \
|
|
||||||
"http://${PIHOLE_HOST}/api/config/dns/hosts" >/dev/null ||
|
|
||||||
echo "WARN: pi-hole DNS update failed (non-fatal)"
|
|
||||||
curl -fsS --max-time 5 -X DELETE -H "sid: ${SID}" "http://${PIHOLE_HOST}/api/auth" >/dev/null || true
|
|
||||||
else
|
|
||||||
echo "WARN: could not obtain pi-hole session (skipping DNS)"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 4) Initial backup (best-effort, can take a few minutes)
|
|
||||||
if [[ -n "${BACKUP_STORAGE}" ]]; then
|
|
||||||
if pvesh get "/storage/${BACKUP_STORAGE}" >/dev/null 2>&1; then
|
|
||||||
echo "Triggering initial backup of CTID=${CTID} to ${BACKUP_STORAGE}"
|
|
||||||
vzdump "${CTID}" \
|
|
||||||
--storage "${BACKUP_STORAGE}" \
|
|
||||||
--mode snapshot \
|
|
||||||
--compress zstd \
|
|
||||||
--notes-template "Initial backup of ${APP} (CTID=${CTID})" \
|
|
||||||
--notification-mode auto ||
|
|
||||||
echo "WARN: initial backup failed (non-fatal)"
|
|
||||||
else
|
|
||||||
echo "Backup storage '${BACKUP_STORAGE}' not found — skipping."
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "Post-provision routine complete for ${APP} (CTID=${CTID})"
|
|
||||||
# ▲▲▲ EXAMPLE 3 — END ▲▲▲
|
|
||||||
|
|
||||||
# ============================================================================
|
|
||||||
# ▼▼▼ EXAMPLE 4 — BEGIN ▼▼▼
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
# Name : inject-ssh-and-monitoring.sh
|
|
||||||
# Purpose : Push the host's admin SSH key into the new LXC, install the
|
|
||||||
# Beszel monitoring agent inside the container, and register
|
|
||||||
# an Uptime-Kuma HTTP push monitor for the LXC's IP.
|
|
||||||
# Difficulty : ⭐⭐⭐ Advanced
|
|
||||||
# Requires : pct (host), curl (inside LXC), reachable Beszel hub +
|
|
||||||
# Uptime-Kuma push URL.
|
|
||||||
# ============================================================================
|
|
||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
# --- CONFIG (edit me) -------------------------------------------------------
|
|
||||||
ADMIN_KEY="/root/.ssh/admin_ed25519.pub"
|
|
||||||
BESZEL_HUB_URL="http://192.168.1.10:8090"
|
|
||||||
BESZEL_AGENT_KEY="ssh-ed25519 AAAA... beszel@hub" # public key of the hub
|
|
||||||
UPTIME_KUMA_PUSH_BASE="http://uptime.lan/api/push/abc123"
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
# 1) Inject the admin SSH key
|
|
||||||
if [[ -f "${ADMIN_KEY}" ]]; then
|
|
||||||
echo "Pushing admin SSH key into CTID=${CTID}"
|
|
||||||
pct exec "${CTID}" -- mkdir -p /root/.ssh
|
|
||||||
pct exec "${CTID}" -- chmod 700 /root/.ssh
|
|
||||||
pct push "${CTID}" "${ADMIN_KEY}" /root/.ssh/authorized_keys
|
|
||||||
pct exec "${CTID}" -- chmod 600 /root/.ssh/authorized_keys
|
|
||||||
else
|
|
||||||
echo "WARN: ${ADMIN_KEY} not found on host — skipping SSH key injection"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 2) Wait for outbound networking inside the CT (max 30 s)
|
|
||||||
echo "Waiting for network inside CTID=${CTID}…"
|
|
||||||
for _ in $(seq 1 30); do
|
|
||||||
if pct exec "${CTID}" -- bash -c 'getent hosts deb.debian.org >/dev/null 2>&1'; then
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
sleep 1
|
|
||||||
done
|
|
||||||
|
|
||||||
# 3) Install Beszel agent inside the LXC
|
|
||||||
echo "Installing Beszel agent inside CTID=${CTID}"
|
|
||||||
pct exec "${CTID}" -- bash -s <<'AGENT_INSTALL' || echo "WARN: Beszel install failed"
|
|
||||||
set -euo pipefail
|
|
||||||
ARCH="$(uname -m)"
|
|
||||||
case "$ARCH" in
|
|
||||||
x86_64) ARCH_TAG=amd64 ;;
|
|
||||||
aarch64) ARCH_TAG=arm64 ;;
|
|
||||||
*) echo "Unsupported arch: $ARCH"; exit 1 ;;
|
|
||||||
esac
|
|
||||||
TMP=$(mktemp -d)
|
|
||||||
cd "$TMP"
|
|
||||||
curl -fsSL "https://github.com/henrygd/beszel/releases/latest/download/beszel-agent_linux_${ARCH_TAG}.tar.gz" \
|
|
||||||
| tar -xz
|
|
||||||
install -m 0755 beszel-agent /usr/local/bin/beszel-agent
|
|
||||||
|
|
||||||
cat >/etc/systemd/system/beszel-agent.service <<UNIT
|
|
||||||
[Unit]
|
|
||||||
Description=Beszel Agent
|
|
||||||
After=network-online.target
|
|
||||||
Wants=network-online.target
|
|
||||||
[Service]
|
|
||||||
Environment="PORT=45876"
|
|
||||||
Environment="KEY=__KEY_PLACEHOLDER__"
|
|
||||||
ExecStart=/usr/local/bin/beszel-agent
|
|
||||||
Restart=always
|
|
||||||
[Install]
|
|
||||||
WantedBy=multi-user.target
|
|
||||||
UNIT
|
|
||||||
AGENT_INSTALL
|
|
||||||
|
|
||||||
# Inject the configured public key into the unit file (avoids quoting hell)
|
|
||||||
pct exec "${CTID}" -- sed -i "s|__KEY_PLACEHOLDER__|${BESZEL_AGENT_KEY}|" \
|
|
||||||
/etc/systemd/system/beszel-agent.service
|
|
||||||
|
|
||||||
pct exec "${CTID}" -- systemctl daemon-reload
|
|
||||||
pct exec "${CTID}" -- systemctl enable --now beszel-agent.service ||
|
|
||||||
echo "WARN: could not start beszel-agent"
|
|
||||||
|
|
||||||
# 4) Register an Uptime-Kuma push monitor (host-side, just sends one ping)
|
|
||||||
echo "Pinging Uptime-Kuma push monitor for ${HN}"
|
|
||||||
curl -fsS --max-time 5 \
|
|
||||||
--get \
|
|
||||||
--data-urlencode "status=up" \
|
|
||||||
--data-urlencode "msg=created by community-scripts" \
|
|
||||||
--data-urlencode "ping=1" \
|
|
||||||
--data-urlencode "label=${HN}" \
|
|
||||||
"${UPTIME_KUMA_PUSH_BASE}" >/dev/null ||
|
|
||||||
echo "WARN: Uptime-Kuma push failed (non-fatal)"
|
|
||||||
|
|
||||||
echo "Provisioned monitoring for ${APP} (CTID=${CTID}, IP=${IP})"
|
|
||||||
# ▲▲▲ EXAMPLE 4 — END ▲▲▲
|
|
||||||
|
|
||||||
# ============================================================================
|
|
||||||
# ▼▼▼ EXAMPLE 5 — BEGIN ▼▼▼
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
# Name : per-app-router.sh
|
|
||||||
# Purpose : Single dispatcher hook that runs different actions
|
|
||||||
# depending on the app being installed (NSAPP). Useful when
|
|
||||||
# you want ONE hook for the whole cluster but distinct
|
|
||||||
# behavior for, e.g., databases vs media services.
|
|
||||||
# Difficulty : ⭐⭐⭐ Advanced
|
|
||||||
# ============================================================================
|
|
||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
# --- CONFIG (edit me) -------------------------------------------------------
|
|
||||||
DEFAULT_DNS_SUFFIX="lan"
|
|
||||||
PROM_FILE_SD_DIR="/etc/prometheus/file_sd" # on the host that runs Prometheus
|
|
||||||
# ----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
log() { printf '[%s] %s\n' "$(date +%H:%M:%S)" "$*"; }
|
|
||||||
|
|
||||||
# ---------- shared helpers --------------------------------------------------
|
|
||||||
register_prometheus_target() {
|
|
||||||
local job="$1" port="$2"
|
|
||||||
local file="${PROM_FILE_SD_DIR}/${job}.json"
|
|
||||||
mkdir -p "${PROM_FILE_SD_DIR}"
|
|
||||||
if [[ ! -f "$file" ]]; then echo "[]" >"$file"; fi
|
|
||||||
python3 - "$file" "${IP}:${port}" "${HN}" "${NSAPP}" <<'PY'
|
|
||||||
import json, sys
|
|
||||||
path, target, hn, app = sys.argv[1:5]
|
|
||||||
data = json.load(open(path))
|
|
||||||
# Avoid duplicates
|
|
||||||
data = [b for b in data if target not in b.get("targets", [])]
|
|
||||||
data.append({"targets": [target], "labels": {"hostname": hn, "app": app}})
|
|
||||||
json.dump(data, open(path, "w"), indent=2)
|
|
||||||
PY
|
|
||||||
log "Registered Prometheus target ${IP}:${port} in ${file}"
|
|
||||||
}
|
|
||||||
|
|
||||||
set_ct_options() {
|
|
||||||
local cores="$1" mem="$2" desc="$3"
|
|
||||||
pct set "${CTID}" --cores "${cores}" --memory "${mem}" || true
|
|
||||||
pct set "${CTID}" --description "${desc}" || true
|
|
||||||
}
|
|
||||||
|
|
||||||
# ---------- per-app dispatch ------------------------------------------------
|
|
||||||
log "Dispatching post-install for NSAPP=${NSAPP} CTID=${CTID}"
|
|
||||||
|
|
||||||
case "${NSAPP}" in
|
|
||||||
|
|
||||||
# ------ Databases ---------------------------------------------------------
|
|
||||||
postgresql | mariadb | mongodb | redis | valkey)
|
|
||||||
log "Database role: bumping resources & adding to backup-critical pool"
|
|
||||||
set_ct_options 4 4096 "DB: ${APP}"
|
|
||||||
pvesh set /pools/db-critical --vms "${CTID}" 2>/dev/null || true
|
|
||||||
register_prometheus_target "${NSAPP}-exporter" 9187
|
|
||||||
;;
|
|
||||||
|
|
||||||
# ------ *arr media stack --------------------------------------------------
|
|
||||||
sonarr | radarr | prowlarr | lidarr | readarr | bazarr)
|
|
||||||
log "Media-arr role: tagging + Sonarr/Radarr API webhook"
|
|
||||||
pct set "${CTID}" --tags "community-script;media;arr-stack" || true
|
|
||||||
curl -fsS --max-time 5 -X POST \
|
|
||||||
"http://media-hub.${DEFAULT_DNS_SUFFIX}/hooks/arr-added" \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
-d "{\"app\":\"${NSAPP}\",\"ctid\":${CTID},\"ip\":\"${IP}\"}" \
|
|
||||||
>/dev/null || log "WARN: media-hub webhook failed"
|
|
||||||
;;
|
|
||||||
|
|
||||||
# ------ Web apps that should sit behind NPM/Traefik ----------------------
|
|
||||||
vaultwarden | paperless-ngx | nextcloud | immich | bookstack)
|
|
||||||
log "Web app role: registering reverse-proxy entry"
|
|
||||||
curl -fsS --max-time 5 -X POST \
|
|
||||||
"http://traefik.${DEFAULT_DNS_SUFFIX}/api/dynamic-add" \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
-d "$(
|
|
||||||
cat <<JSON
|
|
||||||
{
|
|
||||||
"name": "${HN}",
|
|
||||||
"host": "${HN}.${DEFAULT_DNS_SUFFIX}",
|
|
||||||
"backend": "http://${IP}",
|
|
||||||
"app": "${NSAPP}"
|
|
||||||
}
|
|
||||||
JSON
|
|
||||||
)" >/dev/null || log "WARN: traefik registration failed"
|
|
||||||
register_prometheus_target "blackbox-http" 80
|
|
||||||
;;
|
|
||||||
|
|
||||||
# ------ Default fallback --------------------------------------------------
|
|
||||||
*)
|
|
||||||
log "No special handling for ${NSAPP} — applying generic defaults"
|
|
||||||
register_prometheus_target "node-exporter" 9100
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
log "Finished dispatcher for ${APP} (CTID=${CTID})"
|
|
||||||
# ▲▲▲ EXAMPLE 5 — END ▲▲▲
|
|
||||||
|
|
||||||
# ============================================================================
|
|
||||||
# END OF EXAMPLES
|
|
||||||
# ============================================================================
|
|
||||||
Reference in New Issue
Block a user