diff --git a/.github/workflows/script-test.yaml b/.github/workflows/script-test.yaml
new file mode 100644
index 0000000..5c88039
--- /dev/null
+++ b/.github/workflows/script-test.yaml
@@ -0,0 +1,177 @@
+name: Run Scripts on PVE Node for testing
+permissions:
+ pull-requests: write
+on:
+ pull_request_target:
+ branches:
+ - main
+ paths:
+ - 'install/**.sh'
+ - 'ct/**.sh'
+
+jobs:
+ run-install-script:
+ runs-on: pvenode
+ steps:
+ - name: Checkout PR branch
+ uses: actions/checkout@v4
+ with:
+ ref: ${{ github.event.pull_request.head.ref }}
+ repository: ${{ github.event.pull_request.head.repo.full_name }}
+ fetch-depth: 0
+
+ - name: Add Git safe directory
+ run: |
+ git config --global --add safe.directory /__w/ProxmoxVE/ProxmoxVE
+
+ - name: Set up GH_TOKEN
+ env:
+ GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
+ run: |
+ echo "GH_TOKEN=${GH_TOKEN}" >> $GITHUB_ENV
+
+ - name: Get Changed Files
+ run: |
+ CHANGED_FILES=$(gh pr diff ${{ github.event.pull_request.number }} --repo ${{ github.repository }} --name-only)
+ CHANGED_FILES=$(echo "$CHANGED_FILES" | tr '\n' ' ')
+ echo "Changed files: $CHANGED_FILES"
+ echo "SCRIPT=$CHANGED_FILES" >> $GITHUB_ENV
+ env:
+ GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
+
+
+ - name: Get scripts
+ id: check-install-script
+ run: |
+ ALL_FILES=()
+ ADDED_FILES=()
+ for FILE in ${{ env.SCRIPT }}; do
+ if [[ $FILE =~ ^install/.*-install\.sh$ ]] || [[ $FILE =~ ^ct/.*\.sh$ ]]; then
+ STRIPPED_NAME=$(basename "$FILE" | sed 's/-install//' | sed 's/\.sh$//')
+ if [[ ! " ${ADDED_FILES[@]} " =~ " $STRIPPED_NAME " ]]; then
+ ALL_FILES+=("$FILE")
+ ADDED_FILES+=("$STRIPPED_NAME") # Mark this base file as added (without the path)
+ fi
+ fi
+ done
+ ALL_FILES=$(echo "${ALL_FILES[@]}" | xargs)
+ echo "$ALL_FILES"
+ echo "ALL_FILES=$ALL_FILES" >> $GITHUB_ENV
+
+ - name: Run scripts
+ id: run-install
+ continue-on-error: true
+ run: |
+ set +e
+ #run for each files in /ct
+ for FILE in ${{ env.ALL_FILES }}; do
+ STRIPPED_NAME=$(basename "$FILE" | sed 's/-install//' | sed 's/\.sh$//')
+ echo "Running Test for: $STRIPPED_NAME"
+ if grep -E -q 'read\s+-r\s+-p\s+".*"\s+\w+' "$FILE"; then
+ echo "The script contains an interactive prompt. Skipping execution."
+ continue
+ fi
+ if [[ $FILE =~ ^install/.*-install\.sh$ ]]; then
+ CT_SCRIPT="ct/$STRIPPED_NAME.sh"
+ if [[ ! -f $CT_SCRIPT ]]; then
+ echo "No CT script found for $STRIPPED_NAME"
+ ERROR_MSG="No CT script found for $FILE"
+ echo "$ERROR_MSG" > result_$STRIPPED_NAME.log
+ continue
+ fi
+ if grep -E -q 'read\s+-r\s+-p\s+".*"\s+\w+' "install/$STRIPPED_NAME-install.sh"; then
+ echo "The script contains an interactive prompt. Skipping execution."
+ continue
+ fi
+ echo "Found CT script for $STRIPPED_NAME"
+ chmod +x "$CT_SCRIPT"
+ RUNNING_FILE=$CT_SCRIPT
+ elif [[ $FILE =~ ^ct/.*\.sh$ ]]; then
+ INSTALL_SCRIPT="install/$STRIPPED_NAME-install.sh"
+ if [[ ! -f $INSTALL_SCRIPT ]]; then
+ echo "No install script found for $STRIPPED_NAME"
+ ERROR_MSG="No install script found for $FILE"
+ echo "$ERROR_MSG" > result_$STRIPPED_NAME.log
+ continue
+ fi
+ echo "Found install script for $STRIPPED_NAME"
+ chmod +x "$INSTALL_SCRIPT"
+ RUNNING_FILE=$FILE
+ if grep -E -q 'read\s+-r\s+-p\s+".*"\s+\w+' "ct/$STRIPPED_NAME.sh"; then
+ echo "The script contains an interactive prompt. Skipping execution."
+ continue
+ fi
+ fi
+ git remote add community-scripts https://github.com/community-scripts/ProxmoxVE.git
+ git fetch community-scripts
+ rm -f .github/workflows/scripts/app-test/pr-build.func || true
+ rm -f .github/workflows/scripts/app-test/pr-install.func || true
+ rm -f .github/workflows/scripts/app-test/pr-alpine-install.func || true
+ rm -f .github/workflows/scripts/app-test/pr-create-lxc.sh || true
+ git checkout community-scripts/main -- .github/workflows/scripts/app-test/pr-build.func
+ git checkout community-scripts/main -- .github/workflows/scripts/app-test/pr-install.func
+ git checkout community-scripts/main -- .github/workflows/scripts/app-test/pr-alpine-install.func
+ git checkout community-scripts/main -- .github/workflows/scripts/app-test/pr-create-lxc.sh
+ chmod +x $RUNNING_FILE
+ chmod +x .github/workflows/scripts/app-test/pr-create-lxc.sh
+ chmod +x .github/workflows/scripts/app-test/pr-install.func
+ chmod +x .github/workflows/scripts/app-test/pr-alpine-install.func
+ chmod +x .github/workflows/scripts/app-test/pr-build.func
+ sed -i 's|source <(curl -s https://raw.githubusercontent.com/community-scripts/ProxmoxVE/main/misc/build.func)|source .github/workflows/scripts/app-test/pr-build.func|g' "$RUNNING_FILE"
+ echo "Executing $RUNNING_FILE"
+ ERROR_MSG=$(./$RUNNING_FILE 2>&1 > /dev/null)
+ echo "Finished running $FILE"
+ if [ -n "$ERROR_MSG" ]; then
+ echo "ERROR in $STRIPPED_NAME: $ERROR_MSG"
+ echo "$ERROR_MSG" > result_$STRIPPED_NAME.log
+ fi
+ done
+ set -e # Restore exit-on-error
+
+ - name: Cleanup PVE Node
+ run: |
+ containers=$(pct list | tail -n +2 | awk '{print $0 " " $4}' | awk '{print $1}')
+
+ for container_id in $containers; do
+ status=$(pct status $container_id | awk '{print $2}')
+ if [[ $status == "running" ]]; then
+ pct stop $container_id
+ pct destroy $container_id
+ fi
+ done
+
+ - name: Post error comments
+ run: |
+ ERROR="false"
+ SEARCH_LINE=".github/workflows/scripts/app-test/pr-build.func: line 255:"
+
+ # Get all existing comments on the PR
+ EXISTING_COMMENTS=$(gh pr view ${{ github.event.pull_request.number }} --repo ${{ github.repository }} --json comments --jq '.comments[].body')
+
+ for FILE in ${{ env.ALL_FILES }}; do
+ STRIPPED_NAME=$(basename "$FILE" | sed 's/-install//' | sed 's/\.sh$//')
+ if [[ ! -f result_$STRIPPED_NAME.log ]]; then
+ continue
+ fi
+ ERROR_MSG=$(cat result_$STRIPPED_NAME.log)
+
+ if [ -n "$ERROR_MSG" ]; then
+ CLEANED_ERROR_MSG=$(echo "$ERROR_MSG" | sed "s|$SEARCH_LINE.*||")
+ COMMENT_BODY=":warning: The script _**$FILE**_ failed with the following message: