prepare release #683
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#################################################################################### | |
# GitHub Action: | |
# Whenever creating a new release of the source code, | |
# also create a release of the installable plugin, | |
# and downgrading it from PHP 8.1 to 7.4 to support more users. | |
#################################################################################### | |
# Steps to execute: | |
# - Checkout the source code | |
# - Run "composer install" for development, to install dependencies for Rector | |
# - Run Rector to downgrade code from PHP 8.1 to 7.4 | |
# - Run "composer install" for production (required dependencies are already under vendor/) | |
# - Create a .zip file, excluding: | |
# - All hidden files (.git, .gitignore, etc) | |
# - Rector file | |
# - All development files, ending in .dist | |
# - All composer files <= after installing dependencies, no need for them anymore | |
# - Markdown files concerning development | |
# - Folder build/ <= created only to store the .zip file | |
# - Folder dev-helpers/ <= not needed for the plugin | |
# - Upload the .zip file as an artifact to the action (this step is possibly optional) | |
# - Upload the .zip file as a release, for download | |
# | |
# You can filter what .zip files to generate by type, by setting variable | |
# `GENERATE_PLUGINS_FILTER` with any of these values: | |
# - extension | |
# - bundle | |
# - standalone-plugin | |
#################################################################################### | |
name: Generate plugins | |
on: | |
release: | |
types: [published] | |
push: | |
branches: | |
- main | |
- versions/* | |
pull_request: null | |
env: | |
CHECKOUT_SUBMODULES: "recursive" | |
GENERATE_PLUGINS_FILTER: "" | |
# see https://github.com/composer/composer/issues/9368#issuecomment-718112361 | |
COMPOSER_ROOT_VERSION: dev-main | |
jobs: | |
provide_data: | |
name: Provide configuration to generate plugins | |
runs-on: ubuntu-latest | |
steps: | |
- uses: actions/checkout@v4 | |
with: | |
submodules: ${{ env.CHECKOUT_SUBMODULES }} | |
- uses: shivammathur/setup-php@v2 | |
with: | |
php-version: 8.1 | |
coverage: none | |
env: | |
COMPOSER_TOKEN: ${{ secrets.GITHUB_TOKEN }} | |
- uses: ramsey/composer-install@v3 | |
- id: output_data | |
run: | | |
quote=\' | |
filter_arg="--filter=$(echo '${{ env.GENERATE_PLUGINS_FILTER }}' | sed -e 's/ / --filter=/g')" | |
echo "plugin_config_entries=$(vendor/bin/monorepo-builder plugin-config-entries-json $(echo $filter_arg) --config=config/monorepo-builder/plugin-config-entries-json.php)" >> $GITHUB_OUTPUT | |
echo "retention_days_for_generated_plugins=$(vendor/bin/monorepo-builder env-var RETENTION_DAYS_FOR_GENERATED_PLUGINS --config=config/monorepo-builder/env-var.php)" >> $GITHUB_OUTPUT | |
echo "local_package_owners=$(vendor/bin/monorepo-builder local-package-owners --config=config/monorepo-builder/local-package-owners.php)" >> $GITHUB_OUTPUT | |
echo "git_base_branch=$(vendor/bin/monorepo-builder env-var GIT_BASE_BRANCH --config=config/monorepo-builder/env-var.php)" >> $GITHUB_OUTPUT | |
echo "git_user_name=$(vendor/bin/monorepo-builder env-var GIT_USER_NAME --config=config/monorepo-builder/env-var.php)" >> $GITHUB_OUTPUT | |
echo "git_user_email=$(vendor/bin/monorepo-builder env-var GIT_USER_EMAIL --config=config/monorepo-builder/env-var.php)" >> $GITHUB_OUTPUT | |
outputs: | |
plugin_config_entries: ${{ steps.output_data.outputs.plugin_config_entries }} | |
retention_days: ${{ steps.output_data.outputs.retention_days_for_generated_plugins }} | |
local_package_owners: ${{ steps.output_data.outputs.local_package_owners }} | |
git_base_branch: ${{ steps.output_data.outputs.git_base_branch }} | |
git_user_name: ${{ steps.output_data.outputs.git_user_name }} | |
git_user_email: ${{ steps.output_data.outputs.git_user_email }} | |
# Build plugin => downgrade => (maybe) scope => (maybe) upload to release and deploy to dist repo | |
process: | |
name: Generate plugin "${{ matrix.pluginConfig.plugin_slug }}" | |
needs: provide_data | |
runs-on: ubuntu-latest | |
strategy: | |
fail-fast: false | |
matrix: | |
pluginConfig: ${{ fromJson(needs.provide_data.outputs.plugin_config_entries) }} | |
steps: | |
- name: Checkout code | |
uses: actions/checkout@v4 | |
with: | |
submodules: ${{ env.CHECKOUT_SUBMODULES }} | |
- name: Create build folder | |
run: mkdir build && mkdir build/dist-plugin | |
- name: Install zip | |
uses: montudor/[email protected] | |
# pcre.jit=0 => @see https://github.com/composer/composer/issues/9595 | |
- name: Use PHP 8.1 | |
uses: shivammathur/setup-php@v2 | |
with: | |
php-version: 8.1 | |
coverage: none | |
ini-values: pcre.jit=0 | |
env: | |
COMPOSER_TOKEN: ${{ secrets.GITHUB_TOKEN }} | |
- name: Install root dependencies | |
uses: ramsey/composer-install@v3 | |
########################################################################### | |
# Downgrade plugin | |
########################################################################### | |
# "custom-bump-interdependency" temporarily needed because of bug: | |
# https://github.com/symplify/symplify/issues/2773 | |
- name: Localize package paths | |
run: | | |
vendor/bin/monorepo-builder custom-bump-interdependency --config=config/monorepo-builder/custom-bump-interdependency.php "dev-${{ needs.provide_data.outputs.git_base_branch }}" | |
vendor/bin/monorepo-builder localize-composer-paths --config=config/monorepo-builder/localize-composer-paths.php ${{ matrix.pluginConfig.path }}/composer.json --ansi | |
########################################################################### | |
# When building bundles (i.e. containing 2 or more extension plugins), if 2 extensions | |
# contain the same entry under "replace" in composer.json, then "require"ing both of | |
# them in the bundle fails. | |
# | |
# For instance, several plugins replace "pop-schema/schema-commons", producing | |
# the following error message: | |
# | |
# > Only one of these can be installed: pop-schema/schema-commons[dev-master], gatographql-pro/php-constants-and-environment-variables-via-schema[dev-master], gatographql-pro/helper-function-collection[dev-master]. [gatographql-pro/php-constants-and-environment-variables-via-schema, gatographql-pro/helper-function-collection] replace pop-schema/schema-commons and thus cannot coexist with it. | |
# | |
# As a solution, when generating the bundle plugin, remove all the "replace" entries | |
# in the composer.json for the included plugins, and move them to the bundle composer.json | |
########################################################################### | |
- name: "Bundles: Transfer the 'replace' entries in composer.json, from the contained plugins to the bundle" | |
run: | | |
vendor/bin/monorepo-builder transfer-composer-replace-entries-from-plugins-to-bundle --config=config/monorepo-builder/transfer-composer-replace-entries-from-plugins-to-bundle.php "${{ matrix.pluginConfig.path }}/composer.json" --exclude-replace="${{ matrix.pluginConfig.exclude_replace }}" | |
if: ${{ matrix.pluginConfig.is_bundle }} | |
########################################################################### | |
# When building standalone plugins, the "replace" entries from the bundled | |
# extensions must be ignored, as these must also be contained within the | |
# standalone plugin (which also contains Gato GraphQL). | |
# | |
# Because standalone plugins are bundles, in the previous step the | |
# "replace" entries have been moved up from the bundled extensions | |
# to the bundle's composer.json. Now, remove them. | |
########################################################################### | |
- name: "Standalone plugins: Remove the 'replace' entries in composer.json (originally from the contained plugins, now in the bundle)" | |
run: | | |
vendor/bin/monorepo-builder remove-composer-replace-entries --config=config/monorepo-builder/remove-composer-replace-entries.php "${{ matrix.pluginConfig.path }}/composer.json" | |
if: ${{ matrix.pluginConfig.is_standalone_plugin }} | |
- name: Install plugin dependencies, avoiding v2 platform check | |
run: | | |
composer config platform-check false --no-interaction --ansi | |
composer install --no-progress --no-interaction --ansi | |
working-directory: ${{ matrix.pluginConfig.path }} | |
# before_downgrade_code.sh => Hacks to fix the codebase in preparation of the Rector downgrade | |
- name: Custom bash script to fix items in the code in preparation for the Rector downgrade | |
run: "$GITHUB_WORKSPACE/${{ matrix.pluginConfig.bashScripts.before_downgrade_code }}" | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.bashScripts.before_downgrade_code }} | |
# additional_rector_after_configs => Hack to fix bug: https://github.com/rectorphp/rector/issues/5962 | |
- name: Downgrade code for production (to PHP 7.4) | |
run: submodules/GatoGraphQL/ci/downgrade/downgrade_code.sh "${{ matrix.pluginConfig.rector_downgrade_config }}" "" "${{ matrix.pluginConfig.path }}" "${{ matrix.pluginConfig.additional_rector_before_configs }}" "${{ matrix.pluginConfig.additional_rector_after_configs }}" "${{ needs.provide_data.outputs.local_package_owners }}" | |
# after_downgrade_code.sh => Hacks to fix the codebase whenever Rector cannot handle it | |
- name: Custom bash script to fix items in the code that Rector cannot handle | |
run: "$GITHUB_WORKSPACE/${{ matrix.pluginConfig.bashScripts.after_downgrade_code }}" | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.bashScripts.after_downgrade_code }} | |
################################################################################ | |
- name: Replace PHP version in plugin main file | |
run: | | |
sed -i 's/Requires PHP: 8.1/Requires PHP: 7.4/' ${{ matrix.pluginConfig.main_file }} | |
working-directory: ${{ matrix.pluginConfig.path }} | |
- name: Check if readme.txt exists | |
uses: andstor/file-existence-action@v3 | |
id: check_readme_exists | |
with: | |
files: "${{ matrix.pluginConfig.path }}/readme.txt" | |
- name: Replace PHP version in plugin readme file | |
run: | | |
sed -i 's/Requires PHP: 8.1/Requires PHP: 7.4/' readme.txt | |
if: steps.check_readme_exists.outputs.files_exists == 'true' | |
working-directory: ${{ matrix.pluginConfig.path }} | |
# Add the commit hash to the plugin version, to regenerate the container when testing the generated plugin | |
- name: Append the the commit hash to the plugin/extension version | |
run: | | |
sed -i "s/$commitHash = null;/$commitHash = '${{ github.sha }}';/" ${{ matrix.pluginConfig.main_file }} | |
working-directory: ${{ matrix.pluginConfig.path }} | |
- name: Build project for production | |
run: composer install --no-dev --optimize-autoloader --no-progress --no-interaction --ansi | |
working-directory: ${{ matrix.pluginConfig.path }} | |
########################################################################### | |
# Scope plugin | |
# Only execute when enabled by configuration | |
########################################################################### | |
- name: Install PHP-Scoper | |
run: | | |
composer global config minimum-stability dev | |
composer global config prefer-stable true | |
composer global require humbug/php-scoper | |
if: ${{ matrix.pluginConfig.scoping }} | |
# (Current situation) If the scoped results correspond to vendor/ only, we must do "--output-dir ../prefixed-plugin/vendor" | |
# (Not happening now) If they also include src/, we must do "--output-dir ../prefixed-plugin" | |
- name: Scope code for 3rd-party dependencies into separate folder | |
run: ~/.composer/vendor/bin/php-scoper add-prefix --config=${{ matrix.pluginConfig.scoping.phpscoper_config.external }} --output-dir $GITHUB_WORKSPACE/build/prefixed-plugin/vendor --ansi --no-interaction | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.scoping.phpscoper_config.external }} | |
- name: Copy scoped 3rd-party dependencies code back to source folder | |
run: rsync -av build/prefixed-plugin/ ${{ matrix.pluginConfig.path }} --quiet | |
if: ${{ matrix.pluginConfig.scoping.phpscoper_config.external }} | |
# (Optional) Also scope own classes (eg: for creating a standalone plugin) | |
- name: Scope own code into separate folder | |
run: ~/.composer/vendor/bin/php-scoper add-prefix --config=${{ matrix.pluginConfig.scoping.phpscoper_config.internal }} --output-dir $GITHUB_WORKSPACE/build/prefixed-plugin-internal --ansi --no-interaction | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.scoping.phpscoper_config.internal }} | |
- name: Copy scoped own code back to source folder | |
run: rsync -av build/prefixed-plugin-internal/ ${{ matrix.pluginConfig.path }} --quiet | |
if: ${{ matrix.pluginConfig.scoping.phpscoper_config.internal }} | |
- name: Regenerate autoloader | |
run: composer dumpautoload --optimize --classmap-authoritative --ansi | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.scoping }} | |
- name: Use Scoper autoload in plugin main file | |
run: | | |
sed -i 's/autoload.php/scoper-autoload.php/' ${{ matrix.pluginConfig.main_file }} | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.scoping }} | |
- name: Remove all function aliases from scoper-autoload.php | |
run: | | |
sed -i -E 's/(if \(\!function_exists\(.+\)\) \{ function)/\/\/ \1/' vendor/scoper-autoload.php | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.scoping }} | |
########################################################################### | |
# Generate plugin, and Upload as artifact | |
########################################################################### | |
- name: Create plugin as zip | |
run: zip -X -r $GITHUB_WORKSPACE/build/${{ matrix.pluginConfig.zip_file }}.zip . -x *.git* node_modules/\* .* "*/\.*" *.md phpstan.neon *.dist composer.* vendor/**/phpstan.neon vendor/**/phpstan.neon.dist vendor/**/phpunit.xml.dist vendor/**/composer.json vendor/**/README.md vendor/**/LICENSE.md vendor/**/CHANGELOG.md tests/\* **/tests/\* ${{ matrix.pluginConfig.exclude_files }} | |
working-directory: ${{ matrix.pluginConfig.path }} | |
- name: Uncompress plugin zip contents into new folder | |
uses: montudor/[email protected] | |
with: | |
args: unzip -qq build/${{ matrix.pluginConfig.zip_file }}.zip -d build/dist-plugin/${{ matrix.pluginConfig.plugin_slug }} | |
- name: Upload plugin zip as artifact | |
uses: actions/upload-artifact@v4 | |
with: | |
name: ${{ matrix.pluginConfig.zip_file }} | |
path: build/dist-plugin/ | |
retention-days: ${{ needs.provide_data.outputs.retention_days }} | |
########################################################################### | |
# Upload and Deploy | |
# Only when doing a release | |
########################################################################### | |
- name: Create release folder | |
run: mkdir build/release-plugin | |
if: github.event_name == 'release' | |
- name: Create release plugin as .zip file (containing a root folder with the plugin name) | |
run: zip -X -r $GITHUB_WORKSPACE/build/release-plugin/${{ matrix.pluginConfig.zip_file }}.zip . | |
working-directory: build/dist-plugin | |
if: github.event_name == 'release' | |
- name: Upload to release | |
uses: softprops/action-gh-release@v2 | |
with: | |
files: build/release-plugin/${{ matrix.pluginConfig.zip_file }}.zip | |
env: | |
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | |
if: github.event_name == 'release' | |
- id: previous_tag | |
uses: WyriHaximus/github-action-get-previous-tag@master | |
if: github.event_name == 'release' | |
- name: Include (previously excluded) folders for DIST repo | |
run: sudo rsync -av ${{ matrix.pluginConfig.include_folders_for_dist_repo }} $GITHUB_WORKSPACE/build/dist-plugin/${{ matrix.pluginConfig.plugin_slug }} --quiet | |
working-directory: ${{ matrix.pluginConfig.path }} | |
if: ${{ matrix.pluginConfig.include_folders_for_dist_repo != '' && github.event_name == 'release' && matrix.pluginConfig.dist_repo_organization && matrix.pluginConfig.dist_repo_name }} | |
- name: Publish to DIST repo | |
uses: symplify/[email protected] | |
env: | |
GITHUB_TOKEN: ${{ secrets.ACCESS_TOKEN }} | |
with: | |
branch: ${{ matrix.pluginConfig.dist_repo_branch }} | |
package-directory: 'build/dist-plugin/${{ matrix.pluginConfig.plugin_slug }}' | |
split-repository-organization: ${{ matrix.pluginConfig.dist_repo_organization }} | |
split-repository-name: ${{ matrix.pluginConfig.dist_repo_name }} | |
tag: ${{ steps.previous_tag.outputs.tag }} | |
user-name: "${{ needs.provide_data.outputs.git_user_name }}" | |
user-email: "${{ needs.provide_data.outputs.git_user_email }}" | |
if: ${{ github.event_name == 'release' && matrix.pluginConfig.dist_repo_organization && matrix.pluginConfig.dist_repo_name }} | |