Harness Base PHP for Workspace
A Workspace harness is a way to ship files to a project without being part of the project.
In this repository are a set of harnesses that have been created for the PHP language, reducing the maintenance overhead of the individual harnesses greatly. This is due to the "base" PHP harness being used as a common set of templates.
Each framework will fully override a base harness file if differing behaviour is required.
- PHP (used as a base for everything else), published to inviqa/harness-php
- Akeneo (PIM), published to inviqa/harness-akeneo
- Drupal 8 (CMS), published to inviqa/harness-drupal8
- Magento 1.x (Ecommerce), published to inviqa/harness-magento1
- Magento 2.x (Ecommerce), published to inviqa/harness-magento2
- Spryker (Ecommerce), published to inviqa/harness-spryker
- Symfony, published to inviqa/harness-symfony
- Wordpress (CMS), published to inviqa/harness-wordpress
- Local docker-compose development environment
- Skeleton for simple set-up of new projects
- Pipeline docker-compose environment for use in Jenkins or other tools to run tests
- Helm chart for deploying QA, UAT and Production environments to Kubernetes clusters
A developer for a project can follow these steps to upgrade their harness version:
- Check the version specific upgrade instructions for the new version to see if there are any specific steps. If upgrading through multiple versions, check the previous version's instructions too!
- Check what harness is in use for the project. This is usually line 2 or 3 in
workspace.yml
. - Clone the repository for the harness in use for the project and check out the new tagged version. For example:
git clone git@github.com:inviqa/harness-<framework>.git
cd harness-<framework>
git checkout 1.1.1
pwd # Use this path for the diff in step 3
- Update the
workspace.yml
harness version (usually line 2 or 3) to the new tagged version. - Perform a recursive diff between the new release from the checked out harness from step 1 and any "overlay"
directories such as
tools/workspace/
.- Remove any files from the project that are now the same as the harness.
- Port over any changes from the harness to override files that must stay.
- Download the new harness version and render the twig templates:
rm -rf .my127ws
ws harness download
ws harness prepare
- Perform a recursive diff between the
.my127ws/application/skeleton/
folder to the project root:- If a skeleton file (that isn't in a
_twig
folder or named*.twig
) is missing, copy it to the project - If a project file is missing some changes from the skeleton, try applying the change from the skeleton.
- Port over new features such as new standard dev tooling in composer.json.
- Ensure the README.md is up to date, though keep any changes made to the project's README.md over time.
- If a skeleton file (that isn't in a
- Compare the overrides for attributes in the project's
workspace.yml
to the harness'sharness.yml
andharness/attributes/*.yml
.- Port over any additional build, init or migrate steps.
- Remove any attribute overrides from workspace.yml that are now the same as the harness.
- Test with
ws harness update existing
- Open a pull request with the project and ensure CI checks (such as Jenkins) pass.
- Note on the pull request that to apply the changes, the following needs to be done by the project team:
To keep your existing database:
`ws harness update existing`
To do a fresh installation:
`ws harness update fresh`
- Ask for someone else to test the pull request.
- Once the pull request has been merged to the default branch of the project, remind the project team to apply the changes with the
ws harness update existing
orws harness update fresh
commands.
Each harness deploys:
- A "console" pod for running one-off commands
- A NGINX/PHP-FPM "webapp" pod for running the php-based application and serving web requests
- A service to route to the "webapp" pods
- An ingress definition to route via the "webapp" service
- Optionally, a "cron" pod for running cronjobs
- Optionally, elasticsearch, mysql, postgres, redis for supporting services
The memory requests for pods have been deliberately set to be the same as the limits.
This is to avoid nodes going to "NotReady" status due to dockerd/containerd/kubelet being killed by the kernel.
An example: Requesting 10Mi of memory but allowing the pod to spike to 1024Mi means that kubernetes will schedule the pod it onto a node with 10Mi allocatable memory left. It doesn't consider the limits at all when scheduling pods.
As soon as something in the pod starts using more than 10Mi when the node is already at capacity, kubelet attempts to kill processes in the container to get back down to 10Mi.
Sometimes kubelet does not manage to kick in fast enough and the Linux kernel's Out Of Memory (OOM) killer kicks in instead. Whilst core kubernetes processes such as dockerd, containerd and kubelet have an extremely low priority for the OOMKiller, sometimes the kernel decides to kill one of these core processes anyway as it would free up the most memory, leading to the node having issues.
The final harness version for each of the frameworks is put together by the build script into a "dist" folder. This is used for testing in Jenkins.
- Run
./build
- Create and change to a tests directory
mkdir tests
cd tests
ws create <framework>-test inviqa/<framework> --no-install
- Copy the built version of harness-base-php's
./dist/<framework>
dir to<framework>-test/.my127ws
directory
cp -pR ../dist/harness-<framework>/ <framework>-test/.my127ws/
- Change directory to
<framework>-test
cd `<framework>-test`
- Update the override file in
<framework>-test
:
echo "attribute('mutagen'): no" >> workspace.override.yml
- Run in
pipeline
mode to activatestatic
mode
MY127WS_ENV=pipeline ws install
The deploy script does a similar thing but the end result is output to src/<framework>
, where it is then
committed as a publish commit to a temporary build branch.
A "subtree-split" is then performed which outputs a directory for each folder into a "publish" folder, where it is then
force pushed to the individual harness repositories' 1.2.x
branch.
We are keeping a changelog, powered by GitHub Changelog Generator.
When ready to tag a release, make a new branch from the 1.2.x
branch for the changelog entries:
- Generate a
repo
scope token for use with the changelog generator: https://github.com/settings/tokens/new?description=GitHub%20Changelog%20Generator%20token - Export it in your environment:
export CHANGELOG_GITHUB_TOKEN=...
- Run the following docker command to generate the changelog, replacing
1.2.0
with the version number as needed:
docker run -e CHANGELOG_GITHUB_TOKEN="$CHANGELOG_GITHUB_TOKEN" -it --rm -v "$(pwd)":/usr/local/src/your-app -v "$(pwd)/github-changelog-http-cache":/tmp/github-changelog-http-cache ferrarimarco/github-changelog-generator --user inviqa --project harness-base-php --exclude-labels "duplicate,question,invalid,wontfix,skip-changelog" --release-branch 1.2.x --future-release 1.2.0
- Examine the generated CHANGELOG.md. For every entry in the
Merged pull requests
section, examine the Pull Requests and assign each pull request either aenhancement
label for a new feature,bug
for a bugfix ordeprecated
for a deprecation. - For each Pull Request in the release, assign an appropriate
harness-*
label. - Re-generate the changelog using step 3 as needed.
- Adjust the version for each framework's README.md:
sed -i '' s/1\.1\.1/1.2.0/ README.md src/*/README.md src/*/docs/*.md src/*/docs/*/*.md
- Commit the resulting changes, push and raise a pull request.
- Once merged, continue with the release process below.
Once the CHANGELOG.markdown is in the branch you wish to release:
- Tag the release version with
git tag 1.2.0 -m "v1.2.0"
- Push the tag to the repository:
git push origin 1.2.0
- Verify you don't have any ignored files in
src/
, and clean up if you do:git status --ignored
- Run the deploy script:
./deploy
- Submit a pull request to my127/my127.io which adds the new release version and asset download URL for the
php-based harnesses to
harnesses.json
- Create a "Github Release" for this repository and downstream repositories, pasting in the changelog for the release:
- https://github.com/inviqa/harness-base-php/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-php/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-akeneo/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-drupal8/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-magento1/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-magento2/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-spryker/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-symfony/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
- https://github.com/inviqa/harness-wordpress/releases/new?tag=1.2.0&title=1.2.0&target=1.2.x
If the next release does not make sense to be in the current 1.2.x branch:
- Create a new branch:
git checkout -b 1.3.x
- Adjust references from 1.2.x to 1.3.x:
grep -FR '1.3.x' . | grep -v dist/
grep -FR '1.2.x' . | grep -v dist/
# Edit resulting files
- Adjust references in this file from 1.2.0 to 1.3.0:
grep -FR '1.3.0' README.md
grep -FR '1.2.0' README.md
# Edit resulting files
- Commit the resulting files and push:
git add -p
git commit
git push origin -u HEAD
- Change the default branch in GitHub settings and re-target any open PRs against the new default branch.
- Run a deployment of the 1.2.x branch so the branches now exist in the downstream repositories:
./deploy
- Adjust each downstream repository's default branch so people can easily follow the README