Skip to content

Updates

Updates #464

Workflow file for this run

---
name: "Updates"
on:
schedule:
- cron: "0 1 */3 * *" # At 01:00 on every 3rd day-of-month.
# push:
# branches: 'drupals'
env:
# Update vars.
UPDATES_BRANCH: "auto-updater"
UPDATE_COMMIT_MSG: "scheduled updates."
# Git.
GITHUB_TOKEN: ${{ secrets.DEVREL_TOKEN }}
GIT_EMAIL: ${{ secrets.DEVREL_EMAIL }}
GIT_USER: ${{ secrets.DEVREL_USER }}
# Notifications.
SLACK_TOKEN: ${{ secrets.SLACK_TOKEN }}
SLACK_CHANNEL_ID: ${{ secrets.SLACK_CHANNEL_ID }}
JOB_ID: "auto-updates"
START_MESSAGE: "Retrieving updates for all templates."
JOB_COLOR: "#145CC6"
# Language-specific tools.
NODE_VERSION: "14"
GO_VERSION: "1.16"
POETRY_VERSION: "1.1.11"
PYTHON_VERSION: "3.9"
RUBY_VERSION: "2.7.4"
BUNDLER_VERSION: "2.2.26"
REPOOWNER: "platformsh"
TEMPLATEOWNER: "platformsh-templates"
#List of all python-based templates
PYTHON_TEMPLATES: "fastapi django2 django3 django4 python3 pyramid python3-uwsgi meilisearch flask wagtail pelican"
jobs:
# A. Get the list of templates that will be used in the update (all subdirectories in `templates/`).
build:
runs-on: ubuntu-latest
name: "Get templates"
outputs:
templates: ${{ steps.getlist.outputs.templates }}
threadts: ${{ steps.startslack.outputs.threadts }}
steps:
- uses: actions/checkout@v2
- name: Setup Python
uses: actions/setup-python@v2
with:
python-version: ${{ env.PYTHON_VERSION }}
- name: "Get template list"
id: getlist
run: |
TEMPLATES=$(python -c '
import os
import json
dirs=os.listdir("{}/templates".format(os.getcwd()))
dirs.remove("__init__.py")
test = {"template": dirs}
print(json.dumps(dirs))
')
echo "templates=${TEMPLATES}" >> $GITHUB_OUTPUT
- name: Install Poetry
run: |
python -m pip install poetry==$POETRY_VERSION
- name: Configure Poetry
run: |
python -m poetry config virtualenvs.in-project true
- name: Cache the virtualenv
uses: actions/cache@v2
with:
path: ./.venv
key: ${{ runner.os }}-venv-${{ hashFiles('**/poetry.lock') }}
- name: Install dependencies
run: |
python -m poetry install
- name: "Start notification on Slack"
id: startslack
run: |
RUN_URL=https://github.com/$REPOOWNER/template-builder/actions/runs/$GITHUB_RUN_ID
THREAD_ID=$(SLACK_BOT_TOKEN=$SLACK_TOKEN CHANNEL_ID=$SLACK_CHANNEL_ID JOB_ID="$JOB_ID" JOB_COLOR=$JOB_COLOR poetry run python utils/slack_notifier.py start $RUN_URL "$START_MESSAGE")
echo "threadts=${THREAD_ID}" >> $GITHUB_OUTPUT
# B. Run the update across the templates list.
update:
needs: build
runs-on: ubuntu-latest
name: "Template"
strategy:
fail-fast: false
matrix:
template: ${{fromJSON(needs.build.outputs.templates)}}
exclude:
# PHP
- template: "magento2ce"
- template: "pimcore"
- template: "laravel"
- template: "typo3"
- template: "sculpin"
- template: "akeneo"
- template: "wordpress-vanilla"
- template: sylius
# Node.js
- template: "directus"
- template: "probot"
################################
# Multi-apps
# Strapis
- template: "strapi"
- template: "eleventy-strapi"
- template: "gatsby-strapi"
# Drupals
- template: "gatsby-drupal"
# WP
- template: "gatsby-wordpress"
# Other
- template: "elastic-apm"
################################
# Java
- template: "microprofile-wildfly"
- template: "microprofile-openliberty"
- template: "microprofile-helidon"
- template: "microprofile-tomee"
- template: "microprofile-kumuluzee"
- template: "microprofile-thorntail"
- template: "microprofile-payara"
- template: "spring-kotlin"
- template: "spring-boot-gradle-mysql"
- template: "spring-boot-maven-mysql"
- template: "spring-mvc-maven-mongodb"
- template: "tomcat"
- template: "jetty"
- template: "jenkins"
- template: "quarkus"
- template: "xwiki"
- template: "micronaut"
# .NET
- template: "aspnet-core"
################################
# Working excludes for testing #
################################
# - template: contentacms
# - template: koa
- template: wagtail
# - template: backdrop
# - template: drupal8
# - template: wordpress-bedrock
# - template: wordpress-composer
# - template: django2
# - template: rails
# - template: echo
# - template: beego
# - template: nodejs
- template: django4
- template: nuxtjs
# - template: drupal8-opigno
# - template: nextcloud
# - template: drupal9-multisite
# - template: mattermost
# - template: gatsby
# - template: hugo
# - template: drupal8-multisite
- template: django3
# - template: lisp
# - template: wordpress-woocommerce
# - template: golang
# - template: express
# - template: strapi4
# - template: nextjs
# - template: meilisearch
# - template: drupal9
- template: flask
# - template: fastapi
- template: pyramid
# - template: contentacms
# - template: gin
# - template: pelican
steps:
# Before beginning we need set up all our tools and dependencies
# 0. Install our tools and dependencies.
- uses: actions/checkout@v2
- name: Pre-Start a. Setup Ruby
uses: ruby/setup-ruby@v1
with:
ruby-version: ${{ env.RUBY_VERSION }}
- name: Pre-Start b. Setup Python
uses: actions/setup-python@v2
with:
python-version: ${{ env.PYTHON_VERSION }}
- name: Install pipenv
run: pip install --user pipenv
- name: Pre-start c. Setup Node
uses: actions/setup-node@v2
with:
node-version: ${{ env.NODE_VERSION }}
- name: Pre-Start d. Setup Go
uses: actions/setup-go@v2
with:
go-version: ${{ env.GO_VERSION }}
- name: Setup PHP
uses: shivammathur/setup-php@v2
with:
php-version: "8.0"
- name: Pre-Start e. Install Poetry
run: |
python -m pip install poetry==$POETRY_VERSION
- name: Pre-Start f. Configure Poetry
run: |
python -m poetry config virtualenvs.in-project true
- name: Pre-Start g. Cache the virtualenv
uses: actions/cache@v2
with:
path: ./.venv
key: ${{ runner.os }}-venv-${{ hashFiles('**/poetry.lock') }}
- name: Pre-Start h. Install dependencies
run: |
python -m poetry install
- name: Pre-Start i. Set default var values
run: |
# according to the docs we should be able to set *and update* env vars to pass data between steps
# @see https://docs.github.com/en/actions/learn-github-actions/workflow-commands-for-github-actions#setting-an-environment-variable
php --version
echo "SLACK_MSG=''" >> $GITHUB_ENV
echo "PR_STATUS=continue" >> $GITHUB_ENV
##########
# Steps 1 - 5 need to happen regardless if we are going to do an update as we need them to be performed for every
# template repository
###########
# 1. Make sure the template repo exists
- name: Step 1. Verify the template repo exists
id: template-repo-exists
run: |
# echo "::set-output name=slack_runurl::https://github.com/${REPOOWNER}/template-builder/actions/runs/${GITHUB_RUN_ID}"
# Why curl instead of the gh cli? Because on not found, the gh cli will return an exit code of 1 which immediately fails the step and the job
repoData=$(curl -s -H "Authorization: token $GITHUB_TOKEN" https://api.github.com/repos/${TEMPLATEOWNER}/${{ matrix.template }})
repoID=$(echo "$repoData" | jq '.id')
repoNodeID=$(echo "$repoData" | jq '.node_id')
repoFound=$(echo "$repoData" | jq '.message')
echo "::notice::Repo Found status - ${repoFound}"
if [[ -z ${repoID} || "${repoFound}" == '"Not Found"' ]]; then
message="The template repo for ${{ matrix.template }} doesn't exist and will need to be created before I can help you."
echo "::error::${message}"
echo "SLACK_MSG=${message}" >> $GITHUB_ENV
echo "SLACK_PRURL=no-pr" >> $GITHUB_ENV
echo "SLACK_STATUS=skip" >> $GITHUB_ENV
exit 11;
else
echo "repoNodeID=${repoNodeID}" >> $GITHUB_OUTPUT
fi
# 2. Verify the repository is set for automerge
- name: Step 2. Check if auto-merge is enabled for repository
run: |
autoMREnabled=$(gh api /repos/${TEMPLATEOWNER}/${{ matrix.template }} --jq '.allow_auto_merge')
if [[ "${autoMREnabled}" != "true" ]]; then
gh api -X PATCH /repos/${TEMPLATEOWNER}/${{ matrix.template }} --field allow_auto_merge=true
fi
#3. Get the template repo's default branch
- name: Step 3. Get template default branch
id: defaultbranch
run: |
# Get the default branch.
DEFAULT_BRANCH=$(gh api /repos/${TEMPLATEOWNER}/${{ matrix.template }} --jq '.default_branch')
echo "Our default branch for this template is ${DEFAULT_BRANCH}"
echo "branch=${DEFAULT_BRANCH}" >> $GITHUB_OUTPUT
# 4. Create branch protection if does not exist on the default branch; update to latest version.
- name: Prepare Repo for Auto Merge
id: prepare-repo
uses: platformsh/prep-for-autopr@main
with:
github-token: ${{ secrets.DEVREL_TOKEN }}
repo-owner: "${TEMPLATEOWNER}"
repo-name: ${{ matrix.template }}
status-checks: ""
# 5. Run the update tasks through template-builder.
- name: "Step 5. Cleanup template build area"
run: |
python -m poetry run doit cleanup:${{ matrix.template }}
# 6. Initialize template build.
- name: "Step 6. Initialize template build"
uses: actions/checkout@v2
with:
token: ${{ secrets.DEVREL_TOKEN }}
repository: ${{ env.TEMPLATEOWNER }}/${{ matrix.template }}
path: templates/${{ matrix.template }}/build
# 7. Handle an existing auto-updater branch.
- name: Step 7. Check if update branch exists on GH
id: branchstatus
run: |
STATUS=$(curl -s -H "Authorization: token $GITHUB_TOKEN" https://api.github.com/repos/${TEMPLATEOWNER}/${{ matrix.template }}/branches/$UPDATES_BRANCH)
branchStatus=$( echo $STATUS | jq -r '.message' )
if [[ 'Branch not found' != "$branchStatus" ]]; then
# make sure our local repository is up-to-date
git -C templates/${{ matrix.template }}/build fetch origin
# our auto-updater branch exists, but has it already been merged and just not deleted?
defaultBranch=${{ steps.defaultbranch.outputs.branch }}
mergeBase=$(git -C templates/${{ matrix.template }}/build merge-base "origin/${defaultBranch}" "origin/${UPDATES_BRANCH}")
# git rev-parse origin/auto-updater
updateBranchHash=$(git -C templates/${{ matrix.template }}/build rev-parse "origin/${UPDATES_BRANCH}")
if [[ "${mergeBase}" != "${updateBranchHash}" ]]; then
message="Update branch already exists and does not appear to have been merged, or merged and then advanced. Skipping template until investigation on open branch (https://github.com/${TEMPLATEOWNER}/${{ matrix.template }}/tree/$UPDATES_BRANCH) is closed."
echo "::error::${message}"
echo "SLACK_MSG=${message}" >> $GITHUB_ENV
echo "SLACK_PRURL=no-pr" >> $GITHUB_ENV
echo "SLACK_STATUS=skip" >> $GITHUB_ENV
exit 12; # should fail the remaining steps
fi
fi
echo "status=$( echo $STATUS | jq -r '.message' )" >> $GITHUB_OUTPUT
- name: "Step 8. Initialize remote"
run: |
# Get the remote
export TEMPLATE=${{ matrix.template }}
REMOTE_UPSTREAM=$(poetry run python -c '
import os
import dodo
try:
print(dodo.project_factory(os.environ["TEMPLATE"]).remote)
except:
print("no remote")
')
# Set up Git.
cd templates/${{ matrix.template }}/build
git config --global user.email "$GIT_EMAIL"
git config --global user.name "$GIT_USER"
if [ "$REMOTE_UPSTREAM" == "no remote" ]; then
echo "$TEMPLATE is type Basic. Skipping remote clone."
else
echo "Upstream found: $REMOTE_UPSTREAM"
git remote add project $REMOTE_UPSTREAM
fi
- name: "Step 9. Update template"
run: |
# For some reason, in the template-builder app, on first run, python needs a second update before it actually updates
UPDATES_BRANCH=$UPDATES_BRANCH UPDATE_COMMIT_MSG=$UPDATE_COMMIT_MSG python -m poetry run doit update:${{ matrix.template }}
if [[ ${PYTHON_TEMPLATES} =~ ${{ matrix.template }} ]]; then
UPDATES_BRANCH=$UPDATES_BRANCH UPDATE_COMMIT_MSG=$UPDATE_COMMIT_MSG python -m poetry run doit update:${{ matrix.template }}
fi
- name: "Step 10. Platformify template"
run: |
UPDATES_BRANCH=$UPDATES_BRANCH UPDATE_COMMIT_MSG=$UPDATE_COMMIT_MSG python -m poetry run doit platformify:${{ matrix.template }}
- name: "Step 11. Branch template"
run: |
UPDATES_BRANCH=$UPDATES_BRANCH UPDATE_COMMIT_MSG=$UPDATE_COMMIT_MSG python -m poetry run doit branch:${{ matrix.template }}
- name: "Step 12. Push template"
run: |
UPDATES_BRANCH=$UPDATES_BRANCH UPDATE_COMMIT_MSG=$UPDATE_COMMIT_MSG python -m poetry run doit push:${{ matrix.template }}
# 13. Check to see updates have been pushed to a branch.
- name: Step 13. Check if updates have been pushed
id: pushstatus
run: |
STATUS=$(curl -s -H "Authorization: token $GITHUB_TOKEN" https://api.github.com/repos/${TEMPLATEOWNER}/${{ matrix.template }}/branches/$UPDATES_BRANCH)
step11=$( echo $STATUS | jq -r '.message' )
echo "pushed branch status is ${step11}"
if [[ "${step11}" = "Branch not found" ]]; then
message="The update branch ${UPDATES_BRANCH} for the template ${{ matrix.template }} was not found. Can't create a PR without it."
message="${message} Possibly the template had no updates."
echo "::error::${message}"
echo "SLACK_MSG=${message}" >> $GITHUB_ENV
echo "SLACK_PRURL=no-pr" >> $GITHUB_ENV
echo "SLACK_STATUS=skip" >> $GITHUB_ENV
echo "PR_STATUS=skip" >> $GITHUB_ENV
echo "status=$( echo $STATUS | jq -r '.message' )" >> >> $GITHUB_OUTPUT
# exit 110; // we dont want to exit as it causes this job to be marked as "failed"
else
echo "status=$( echo $STATUS | jq -r '.message' )" >> $GITHUB_OUTPUT
echo "PR_STATUS=continue" >> $GITHUB_ENV
fi
- name: "Step 14. Open pull request"
id: open-pull-request
if: env.PR_STATUS != 'skip'
run: |
TITLE="Scheduled updates."
BODY="Scheduled updates from template-builder ($(date))."
# If successful, the cli will return the URL to the created PR.
response=$(gh pr create --head "$UPDATES_BRANCH" --base "${{ steps.defaultbranch.outputs.branch }}" --title "$TITLE" --body "$BODY" --repo ${TEMPLATEOWNER}/${{ matrix.template }})
# If the CLI returns 1, the step will fail.
boolPRCreated=$?
prRegex="([0-9]+$)"
if [[ -n "${response}" && $response =~ $prRegex ]]; then
PR=${BASH_REMATCH[1]}
echo "pull-request-link=${response}" >> $GITHUB_OUTPUT
echo "pull-request-number=${PR}" >> $GITHUB_OUTPUT
else
echo "cli didn't fail but response is empty or didnt match the regex?"
echo "${response}"
fi
# 15. Set the pull request to automerge when checks have passed.
- name: Step 15. Automatically merge only after necessary requirements are met
if: env.PR_STATUS != 'skip'
run: |
gh pr merge ${{ steps.open-pull-request.outputs.pull-request-number }} --auto --merge --delete-branch --repo=${TEMPLATEOWNER}/${{ matrix.template }}
lastcommand=$?
# 16. Catch all failures and successes.
- name: Step 16a. Send failed workflows
if: ${{ failure() }}
run: |
TEMPLATE=${{ matrix.template }}
RUN_URL=https://github.com/${REPOOWNER}/template-builder/actions/runs/$GITHUB_RUN_ID
THREAD_ID=${{ needs.build.outputs.threadts }}
PR_URL=${SLACK_PRURL}
MESSAGE="Something went wrong. Investigate.\n${{ env.SLACK_MSG }}"
echo "::warning ::I would send the following warning to slack\n${MESSAGE}"
RESULT=$(SLACK_BOT_TOKEN=$SLACK_TOKEN CHANNEL_ID=$SLACK_CHANNEL_ID THREAD_ID=$THREAD_ID poetry run python utils/slack_notifier.py down $TEMPLATE "$RUN_URL" "$PR_URL" "$MESSAGE")
- name: Step 16b. Send successfull workflows to the pull request
if: ${{ success() && steps.branchstatus.outputs.status == 'Branch not found' }}
run: |
TEMPLATE=${{ matrix.template }}
THREAD_ID=${{ needs.build.outputs.threadts }}
RUN_URL=https://github.com/${REPOOWNER}/template-builder/actions/runs/$GITHUB_RUN_ID
if [ "${{ steps.pushstatus.outputs.status }}" == "Branch not found" ]; then
echo "Template already up-to-date. SKIPPING."
else
PR_URL="<${{ steps.open-pull-request.outputs.pull-request-link}}|View the pull request>"
MESSAGE='All good.'
RESULT=$(SLACK_BOT_TOKEN=$SLACK_TOKEN CHANNEL_ID=$SLACK_CHANNEL_ID THREAD_ID=$THREAD_ID poetry run python utils/slack_notifier.py up $TEMPLATE "$RUN_URL" "$PR_URL" "$MESSAGE")
fi
# C. Notify Slack at the end of the run.
finish:
runs-on: ubuntu-latest
name: "Finish updates"
needs: [build, update]
if: ${{ always() }}
steps:
- uses: actions/checkout@v2
- name: Setup Python
uses: actions/setup-python@v2
with:
python-version: ${{ env.PYTHON_VERSION }}
- name: Install Poetry
run: |
python -m pip install poetry==$POETRY_VERSION
- name: Configure Poetry
run: |
python -m poetry config virtualenvs.in-project true
- name: Cache the virtualenv
uses: actions/cache@v2
with:
path: ./.venv
key: ${{ runner.os }}-venv-${{ hashFiles('**/poetry.lock') }}
- name: Install dependencies
run: |
python -m poetry install
- name: "Notify slack when finished"
run: |
THREAD_ID=${{ needs.build.outputs.threadts }}
RUN_URL=https://github.com/${REPOOWNER}/template-builder/actions/runs/$GITHUB_RUN_ID
RESULT=$(SLACK_BOT_TOKEN=$SLACK_TOKEN CHANNEL_ID=$SLACK_CHANNEL_ID JOB_ID="$JOB_ID" THREAD_ID=$THREAD_ID JOB_COLOR=$JOB_COLOR poetry run python utils/slack_notifier.py finish $RUN_URL "$START_MESSAGE")