Merge pull request #1238 from PHPCSStandards/phpcs-4.x/feature/155-no… #3579
This file contains hidden or 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
name: Test | |
on: | |
# Run on pushes to `master`/`4.x` and on all pull requests. | |
# Prevent the build from running when there are only irrelevant changes. | |
push: | |
branches: | |
- master | |
- 4.x | |
tags: | |
- '**' | |
paths-ignore: | |
- '**.md' | |
pull_request: | |
# Allow manually triggering the workflow. | |
workflow_dispatch: | |
jobs: | |
build: | |
# Cancels all previous runs of this particular job for the same branch that have not yet completed. | |
concurrency: | |
# The concurrency group contains the workflow name, job name and the branch name. | |
group: ${{ github.workflow }}-${{ github.job }}-${{ github.ref }} | |
cancel-in-progress: true | |
name: "Build Phar on PHP: 8.0" | |
permissions: | |
id-token: write | |
contents: read | |
attestations: write | |
uses: ./.github/workflows/reusable-build-phar.yml | |
with: | |
uploadArtifacts: true | |
retentionDays: 28 | |
# Only attests the build artifacts which will be used in the published releases as per the guidelines in "what to attest". | |
# https://docs.github.com/en/actions/security-guides/using-artifact-attestations-to-establish-provenance-for-builds | |
createAttestations: ${{ github.ref_type == 'tag' }} | |
test: | |
# Cancels all previous runs of this particular job for the same branch that have not yet completed. | |
concurrency: | |
# The concurrency group contains the workflow name, job name, job index and the branch name. | |
group: ${{ github.workflow }}-${{ github.job }}-${{ strategy.job-index }}-${{ github.ref }} | |
cancel-in-progress: true | |
runs-on: ${{ matrix.os }} | |
needs: build | |
strategy: | |
# Keys: | |
# - custom_ini: Whether to run with specific custom ini settings to hit very specific | |
# code conditions. | |
matrix: | |
os: ['ubuntu-latest', 'windows-latest'] | |
php: ['7.2', '7.3', '7.4', '8.0', '8.1', '8.2', '8.3', '8.4', '8.5'] | |
custom_ini: [false] | |
include: | |
# Skip test runs on builds which are also run in the coverage job. | |
- php: '7.2' | |
skip_tests: true | |
- php: '8.4' | |
skip_tests: true | |
# Run a couple of builds with custom extensions to allow for testing ini handling within PHPCS. | |
# Ref: https://github.com/PHPCSStandards/PHP_CodeSniffer/issues/416 | |
- php: '7.3' | |
os: 'ubuntu-latest' | |
extensions: ':mysqli' # Run with mysqli disabled. | |
- php: '8.2' | |
os: 'ubuntu-latest' | |
extensions: ':mysqli' # Run with mysqli disabled. | |
# The default libxml library on Ubuntu images is a little out of date. | |
# To safeguard support for the latest libxml we need to update the library on the fly. | |
# This only needs to be tested with one PHP version for each libxml minor to verify support. | |
# Testing against multiple PHP versions would not yield a difference in results. | |
- php: '8.0' | |
os: 'ubuntu-latest' | |
libxml_minor: '2.11' | |
- php: '8.3' | |
os: 'ubuntu-latest' | |
libxml_minor: '2.13' | |
# Extra builds running only the unit tests with different PHP ini settings. | |
- php: '7.4' | |
os: 'ubuntu-latest' | |
custom_ini: true | |
- php: '8.3' | |
os: 'ubuntu-latest' | |
custom_ini: true | |
- php: '8.0' | |
os: 'ubuntu-latest' | |
custom_ini: true | |
- php: '8.2' | |
os: 'ubuntu-latest' | |
custom_ini: true | |
# yamllint disable-line rule:line-length | |
name: "PHP: ${{ matrix.php }} ${{ matrix.custom_ini && ' with custom ini settings' || '' }}${{ matrix.libxml_minor && format( ' with libxml {0}', matrix.libxml_minor ) || '' }} (${{ matrix.os == 'ubuntu-latest' && 'Linux' || 'Win' }})" | |
continue-on-error: ${{ matrix.php == '8.5' }} | |
steps: | |
- name: Prepare git to leave line endings alone | |
run: git config --global core.autocrlf input | |
- name: Checkout code | |
uses: actions/checkout@v5 | |
- name: "libxml2: find the latest relevant tag" | |
if: ${{ matrix.libxml_minor }} | |
id: libxml_version | |
uses: oprypin/find-latest-tag@v1 | |
with: | |
repository: GNOME/libxml2 | |
releases-only: false # The libxml2 repository doesn't use GitHub's "release" feature. | |
prefix: 'v${{ matrix.libxml_minor }}.' # Limit the result to the minor we're interested in. | |
sort-tags: true # Find the "greatest" version for that minor based on semver. | |
# To put it simply: we need to remove the 'v' prefix from the version number. | |
- name: "libxml2: parse the version to a patch version" | |
if: ${{ matrix.libxml_minor }} | |
id: libxml_patch_version | |
shell: bash | |
env: | |
TAG: ${{ steps.libxml_version.outputs.tag }} | |
run: echo "PATCH=$( echo "$TAG" | cut -b 2- )" >> "$GITHUB_OUTPUT" | |
- name: "libxml2: restore cache" | |
if: ${{ matrix.libxml_minor }} | |
id: libxml_cache_restore | |
uses: actions/cache/restore@v4 | |
with: | |
path: "libxml2-${{ steps.libxml_patch_version.outputs.PATCH }}" | |
key: "${{ matrix.os }}-libxml-${{ matrix.libxml_minor }}-${{ steps.libxml_patch_version.outputs.PATCH }}" | |
# Updating the lists can fail intermittently, typically after Microsoft has released a new package. | |
# This should not be blocking for this job, so ignore any errors from this step. | |
# Ref: https://github.com/dotnet/core/issues/4167 | |
- name: "libxml2: Update the available packages list" | |
if: ${{ matrix.libxml_minor && steps.libxml_cache_restore.outputs.cache-hit != 'true' }} | |
continue-on-error: true | |
run: sudo apt-get update | |
- name: "libxml2: Download and build package (linux only)" | |
if: ${{ matrix.libxml_minor && steps.libxml_cache_restore.outputs.cache-hit != 'true' }} | |
env: | |
PATCH: ${{ steps.libxml_patch_version.outputs.PATCH }} | |
run: | | |
sudo apt-get install -y wget build-essential | |
wget "https://download.gnome.org/sources/libxml2/${{ matrix.libxml_minor }}/libxml2-$PATCH.tar.xz" | |
tar -xf "libxml2-$PATCH.tar.xz" | |
cd "libxml2-$PATCH" | |
./configure --prefix=/usr/local | |
make | |
- name: "libxml2: save cache" | |
if: ${{ matrix.libxml_minor && steps.libxml_cache_restore.outputs.cache-hit != 'true' }} | |
id: libxml_cache_save | |
uses: actions/cache/save@v4 | |
with: | |
path: "libxml2-${{ steps.libxml_patch_version.outputs.PATCH }}" | |
key: ${{ steps.libxml_cache_restore.outputs.cache-primary-key }} | |
- name: "libxml2: Install package (linux only)" | |
if: ${{ matrix.libxml_minor }} | |
env: | |
PATCH: ${{ steps.libxml_patch_version.outputs.PATCH }} | |
run: | | |
cd "libxml2-$PATCH" | |
sudo make install | |
sudo ldconfig | |
- name: Setup ini config | |
id: set_ini | |
shell: bash | |
# yamllint disable rule:line-length | |
run: | | |
# Set the "short_open_tag" ini to make sure specific conditions are tested. | |
# Also turn on error_reporting to ensure all notices are shown. | |
if [[ ${{ matrix.custom_ini }} == true ]]; then | |
echo 'PHP_INI=error_reporting=-1, display_errors=On, display_startup_errors=On, date.timezone=Australia/Sydney, short_open_tag=On' >> "$GITHUB_OUTPUT" | |
else | |
echo 'PHP_INI=error_reporting=-1, display_errors=On, display_startup_errors=On' >> "$GITHUB_OUTPUT" | |
fi | |
# yamllint enable rule:line-length | |
- name: Install PHP | |
uses: shivammathur/setup-php@v2 | |
with: | |
php-version: ${{ matrix.php }} | |
ini-values: ${{ steps.set_ini.outputs.PHP_INI }} | |
extensions: ${{ matrix.extensions }} | |
coverage: none | |
- name: "DEBUG: show libxml loaded version (php)" | |
run: php -r 'echo "libxml loaded version = ", LIBXML_LOADED_VERSION, PHP_EOL;' | |
# Install dependencies and handle caching in one go. | |
# @link https://github.com/marketplace/actions/install-php-dependencies-with-composer | |
- name: Install Composer dependencies | |
uses: "ramsey/composer-install@v3" | |
with: | |
composer-options: ${{ matrix.php == '8.5' && '--ignore-platform-req=php+' || '' }} | |
custom-cache-suffix: $(date -u "+%Y-%m") | |
- name: Grab PHPUnit version | |
id: phpunit_version | |
shell: bash | |
# yamllint disable-line rule:line-length | |
run: echo "VERSION=$(php "vendor/bin/phpunit" --version | grep --only-matching --max-count=1 --extended-regexp '\b[0-9]+\.[0-9]+')" >> "$GITHUB_OUTPUT" | |
- name: "DEBUG: Show grabbed version" | |
run: echo ${{ steps.phpunit_version.outputs.VERSION }} | |
- name: Determine PHPUnit config file to use | |
id: phpunit_config | |
shell: bash | |
run: | | |
if [ "${{ startsWith( steps.phpunit_version.outputs.VERSION, '11.' ) }}" == "true" ]; then | |
echo 'FILE=phpunit.xml.dist' >> "$GITHUB_OUTPUT" | |
elif [ "${{ startsWith( steps.phpunit_version.outputs.VERSION, '10.' ) }}" == "true" ]; then | |
echo 'FILE=phpunit.xml.dist' >> "$GITHUB_OUTPUT" | |
else | |
echo 'FILE=phpunit-lte9.xml.dist' >> "$GITHUB_OUTPUT" | |
fi | |
- name: 'PHPCS: set the path to PHP' | |
run: php "bin/phpcs" --config-set php_path php | |
- name: 'PHPUnit: run the full test suite without code coverage' | |
if: ${{ matrix.skip_tests != true }} | |
run: php "vendor/bin/phpunit" -c ${{ steps.phpunit_config.outputs.FILE }} --no-coverage | |
# Do one test run against the complete test suite in CBF mode to ensure all tests can run in CBF mode. | |
- name: 'PHPUnit: run the full test suite without code coverage in CBF mode (PHP 8.3 only)' | |
if: ${{ matrix.php == '8.3' }} | |
run: php "vendor/bin/phpunit" -c ${{ steps.phpunit_config.outputs.FILE }} --exclude-group nothing --no-coverage | |
env: | |
PHP_CODESNIFFER_CBF: '1' | |
- name: 'PHPUnit: run select tests in CBF mode' | |
if: ${{ matrix.skip_tests != true && matrix.php != '8.3' }} | |
run: > | |
php "vendor/bin/phpunit" -c ${{ steps.phpunit_config.outputs.FILE }} --no-coverage | |
--group CBF --exclude-group nothing | |
env: | |
PHP_CODESNIFFER_CBF: '1' | |
# Note: The code style check is run multiple times against every PHP version | |
# as it also acts as an integration test. | |
- name: 'PHPCS: check code style without cache, no parallel' | |
if: ${{ matrix.custom_ini == false }} | |
run: php "bin/phpcs" --no-cache --parallel=1 | |
- name: Download the PHPCS phar | |
if: ${{ matrix.custom_ini == false }} | |
uses: actions/download-artifact@v5 | |
with: | |
name: phpcs-phar | |
# This test specifically tests that the Phar which will be released works correctly on all PHP versions. | |
- name: 'PHPCS: check code style using the Phar file' | |
if: ${{ matrix.custom_ini == false }} | |
run: php phpcs.phar | |
coverage: | |
# Explicitly *NOT* setting "concurrency" for this job to allow for monitoring code coverage for all merges. | |
runs-on: ${{ matrix.os }} | |
strategy: | |
matrix: | |
os: ['ubuntu-latest', 'windows-latest'] | |
# Note: we can only run code coverage builds against PHP 7.2 and PHP 8.4 or higher: | |
# - PHP 7.2 will work as it uses PHPUnit 8, which doesn't use PHP Parser yet. | |
# - As of PHPUnit 9.3 (PHP 7.3+), PHPUnit started using PHP Parser for code coverage, and PHP Parser | |
# also polyfills tokens, but to different (integer) values. | |
# - Additionally, PHP Parser will block PHPUnit from running with a fatal error if it detects | |
# non-integer values for polyfilled tokens.... _sigh_. | |
# - So, aside from PHP 7.2, we can only run code coverage on the last PHP version (or higher) | |
# which added a new token, which is polyfilled by both. | |
# At the time of writing, this means PHP 8.4 or higher. | |
php: ['7.2', '8.4'] | |
custom_ini: [false] | |
include: | |
# Also run one coverage build with custom ini settings for testing the DisallowShortOpenTag sniff. | |
# Also run with a disabled extension for testing the handling of unsettable ini settings by PHPCS. | |
- php: '8.4' | |
os: 'ubuntu-latest' | |
custom_ini: true | |
extensions: ':mysqli' # Run with mysqli disabled. | |
# yamllint disable-line rule:line-length | |
name: "Coverage: ${{ matrix.php }} ${{ matrix.custom_ini && ' with custom ini settings' || '' }} (${{ matrix.os == 'ubuntu-latest' && 'Linux' || 'Win' }})" | |
steps: | |
- name: Prepare git to leave line endings alone | |
run: git config --global core.autocrlf input | |
- name: Checkout code | |
uses: actions/checkout@v5 | |
- name: Setup ini config | |
if: ${{ matrix.custom_ini == true && matrix.os != 'windows-latest' }} | |
id: set_ini | |
shell: bash | |
run: | | |
# Set the "short_open_tag" ini to make sure specific conditions are tested. | |
echo 'PHP_INI=, date.timezone=Australia/Sydney, short_open_tag=On' >> "$GITHUB_OUTPUT" | |
- name: Install PHP | |
uses: shivammathur/setup-php@v2 | |
with: | |
php-version: ${{ matrix.php }} | |
ini-values: error_reporting=-1, display_errors=On, display_startup_errors=On${{ steps.set_ini.outputs.PHP_INI }} | |
extensions: ${{ matrix.extensions }} | |
coverage: xdebug | |
# Install dependencies and handle caching in one go. | |
# @link https://github.com/marketplace/actions/install-php-dependencies-with-composer | |
- name: Install Composer dependencies | |
uses: "ramsey/composer-install@v3" | |
with: | |
# Bust the cache at least once a month - output format: YYYY-MM. | |
custom-cache-suffix: $(date -u "+%Y-%m") | |
- name: Grab PHPUnit version | |
id: phpunit_version | |
shell: bash | |
# yamllint disable-line rule:line-length | |
run: echo "VERSION=$(php "vendor/bin/phpunit" --version | grep --only-matching --max-count=1 --extended-regexp '\b[0-9]+\.[0-9]+')" >> "$GITHUB_OUTPUT" | |
- name: "DEBUG: Show grabbed version" | |
run: echo ${{ steps.phpunit_version.outputs.VERSION }} | |
- name: Determine PHPUnit config file to use | |
id: phpunit_config | |
shell: bash | |
run: | | |
if [ "${{ startsWith( steps.phpunit_version.outputs.VERSION, '11.' ) }}" == "true" ]; then | |
echo 'FILE=phpunit.xml.dist' >> "$GITHUB_OUTPUT" | |
echo 'WARM_CACHE=true' >> "$GITHUB_OUTPUT" | |
elif [ "${{ startsWith( steps.phpunit_version.outputs.VERSION, '10.' ) }}" == "true" ]; then | |
echo 'FILE=phpunit.xml.dist' >> "$GITHUB_OUTPUT" | |
echo 'WARM_CACHE=true' >> "$GITHUB_OUTPUT" | |
elif [ "${{ steps.phpunit_version.outputs.VERSION >= '9.3' }}" == "true" ]; then | |
echo 'FILE=phpunit-lte9.xml.dist' >> "$GITHUB_OUTPUT" | |
echo 'WARM_CACHE=true' >> "$GITHUB_OUTPUT" | |
else | |
echo 'FILE=phpunit-lte9.xml.dist' >> "$GITHUB_OUTPUT" | |
echo 'WARM_CACHE=false' >> "$GITHUB_OUTPUT" | |
fi | |
- name: 'PHPCS: set the path to PHP' | |
run: php "bin/phpcs" --config-set php_path php | |
# PHPUnit 9.3 started using PHP-Parser for code coverage, which can cause issues due to Parser | |
# also polyfilling PHP tokens. | |
# As of PHPUnit 9.3.4, a cache warming option is available. | |
# Using that option prevents issues with PHP-Parser backfilling PHP tokens during our test runs. | |
- name: "Warm the PHPUnit cache (PHPUnit 9.3+)" | |
if: ${{ steps.phpunit_config.outputs.WARM_CACHE == 'true' }} | |
run: > | |
php "vendor/bin/phpunit" -c ${{ steps.phpunit_config.outputs.FILE }} --warm-coverage-cache | |
${{ steps.phpunit_version.outputs.VERSION >= '9.3' && '--coverage-cache ./build/phpunit-cache' || '' }} | |
- name: "Run the unit tests with code coverage" | |
if: ${{ matrix.os != 'windows-latest' }} | |
run: > | |
php "vendor/bin/phpunit" -c ${{ steps.phpunit_config.outputs.FILE }} | |
${{ steps.phpunit_version.outputs.VERSION >= '9.3' && '--coverage-cache ./build/phpunit-cache' || '' }} | |
- name: "Run select tests in CBF mode with code coverage" | |
if: ${{ matrix.os != 'windows-latest' }} | |
run: > | |
php "vendor/bin/phpunit" -c ${{ steps.phpunit_config.outputs.FILE }} | |
${{ steps.phpunit_version.outputs.VERSION >= '9.3' && '--coverage-cache ./build/phpunit-cache' || '' }} | |
--group CBF --exclude-group nothing --coverage-clover build/logs/clover-cbf.xml | |
env: | |
PHP_CODESNIFFER_CBF: '1' | |
- name: "Run the unit tests which may have different outcomes on Windows with code coverage" | |
if: ${{ matrix.os == 'windows-latest' }} | |
run: > | |
php "vendor/bin/phpunit" -c ${{ steps.phpunit_config.outputs.FILE }} | |
${{ steps.phpunit_version.outputs.VERSION >= '9.3' && '--coverage-cache ./build/phpunit-cache' || '' }} | |
--group Windows | |
- name: "Upload coverage results to Coveralls (normal run)" | |
if: ${{ success() }} | |
uses: coverallsapp/github-action@v2 | |
with: | |
format: clover | |
file: build/logs/clover.xml | |
flag-name: os-${{ matrix.os }}-php-${{ matrix.php }}-custom-ini-${{ matrix.custom_ini }} | |
parallel: true | |
- name: "Upload coverage results to Coveralls (CBF run)" | |
if: ${{ matrix.os != 'windows-latest' && success() }} | |
uses: coverallsapp/github-action@v2 | |
with: | |
format: clover | |
file: build/logs/clover-cbf.xml | |
flag-name: cbf-os-${{ matrix.os }}-ubuntu-latest-php-${{ matrix.php }}-custom-ini-${{ matrix.custom_ini }} | |
parallel: true | |
coveralls-finish: | |
needs: coverage | |
if: always() && needs.coverage.result == 'success' | |
runs-on: ubuntu-latest | |
steps: | |
- name: Coveralls Finished | |
uses: coverallsapp/github-action@v2 | |
with: | |
parallel-finished: true |