Author: Sergey Panteleev (saundefined)
Committer: GitHub (web-flow)
Pusher: saundefined
Date: 2024-08-20T09:20:25+03:00

Commit: 
https://github.com/php/web-php/commit/c632d199d1a4916b815113fa81fd5a60deb1e626
Raw diff: 
https://github.com/php/web-php/commit/c632d199d1a4916b815113fa81fd5a60deb1e626.diff

Add visual regression tests (#994)

Co-authored-by: Roman Pronskiy <ro...@pronskiy.com>
Co-authored-by: Derick Rethans <der...@php.net>

Changed paths:
  A  .github/workflows/update-screenshots.yaml
  A  package.json
  A  playwright.config.ts
  A  tests/Visual/SmokeTest.spec.ts
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-archive-1998-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-conferences-index-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-index-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-index-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-php5-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-0-index-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-1-index-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-2-index-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-6-php-chromium.png
  A  
tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-index-php-chromium.png
  A  yarn.lock
  M  .github/workflows/pr-closed.yml
  M  .github/workflows/pr-preview.yml
  M  .gitignore
  M  .htaccess
  M  Makefile


Diff:

diff --git a/.github/workflows/pr-closed.yml b/.github/workflows/pr-closed.yml
index a745b6e0ef..a3823183b2 100644
--- a/.github/workflows/pr-closed.yml
+++ b/.github/workflows/pr-closed.yml
@@ -13,4 +13,6 @@ jobs:
           host: ${{ secrets.PREVIEW_REMOTE_HOST }}
           username: ${{ secrets.PREVIEW_REMOTE_USER }}
           key: ${{ secrets.PREVIEW_SSH_KEY }}
-          script: bash /home/thephpfoundation/scripts/pr_closed.sh web-php ${{ 
github.event.number }}
+          script: |
+            bash /home/thephpfoundation/scripts/pr_closed.sh web-php ${{ 
github.event.number }}
+            bash /home/thephpfoundation/scripts/pr_closed.sh 
web-php-regression-report ${{ github.event.number }}
diff --git a/.github/workflows/pr-preview.yml b/.github/workflows/pr-preview.yml
index 2c62f78f67..8292a59a2a 100644
--- a/.github/workflows/pr-preview.yml
+++ b/.github/workflows/pr-preview.yml
@@ -32,4 +32,99 @@ jobs:
           comment-id: ${{ steps.fc.outputs.comment-id }}
           edit-mode: 'replace'
           body: |
-            🚀 Commit ${{ github.sha }} Deployed on https://web-php-pr-${{ 
github.event.number }}.preview.thephp.foundation
+            🚀 Preview for commit ${{ github.sha }} can be found at 
https://web-php-pr-${{ github.event.number }}.preview.thephp.foundation
+
+  tests_visual:
+    name: "Visual Tests"
+
+    runs-on: "ubuntu-latest"
+    if: github.repository_owner == 'php'
+
+    strategy:
+      matrix:
+        php-version:
+          - "8.2"
+        node-version:
+          - "22.x"
+
+    env:
+      HTTP_HOST: "localhost:8080"
+
+    steps:
+      - name: "Checkout"
+        uses: actions/checkout@v4
+        with:
+          ref: "refs/pull/${{ github.event.number }}/merge"
+
+      - name: "Set up PHP"
+        uses: "shivammathur/setup-php@v2"
+        with:
+          coverage: "none"
+          extensions: "none, curl, dom, json, mbstring, tokenizer, xml, 
xmlwriter"
+          php-version: "${{ matrix.php-version }}"
+
+      - name: Use Node.js ${{ matrix.node-version }}
+        uses: actions/setup-node@v3
+        with:
+          node-version: ${{ matrix.node-version }}
+
+      - name: "Set up problem matchers for PHP"
+        run: "echo \"::add-matcher::${{ runner.tool_cache }}/php.json\""
+
+      - name: "Set up problem matchers for phpunit/phpunit"
+        run: "echo \"::add-matcher::${{ runner.tool_cache }}/phpunit.json\""
+
+      - name: "Determine composer cache directory"
+        run: "echo \"COMPOSER_CACHE_DIR=$(composer config cache-dir)\" >> 
$GITHUB_ENV"
+
+      - name: "Cache dependencies installed with composer"
+        uses: "actions/cache@v3"
+        with:
+          path: "${{ env.COMPOSER_CACHE_DIR }}"
+          key: "php-${{ matrix.php-version }}-composer-${{ 
hashFiles('composer.lock') }}"
+          restore-keys: "php-${{ matrix.php-version }}-composer-"
+
+      - name: "Install dependencies with composer"
+        run: "composer install --ansi --no-interaction --no-progress"
+
+      - name: "Install dependencies"
+        run: "yarn install"
+
+      - name: "Install Playwright"
+        run: "npx playwright install"
+
+      - name: "Run visual tests"
+        run: "make tests_visual"
+
+      - uses: actions/upload-artifact@v4
+        if: ${{ !cancelled() }}
+        with:
+          name: playwright-report
+          path: playwright-report/
+          retention-days: 30
+
+      - uses: easingthemes/ssh-deploy@main
+        if: ${{ !cancelled() }}
+        with:
+          REMOTE_HOST: ${{ secrets.PREVIEW_REMOTE_HOST }}
+          REMOTE_USER: ${{ secrets.PREVIEW_REMOTE_USER }}
+          SSH_PRIVATE_KEY: ${{ secrets.PREVIEW_SSH_KEY }}
+          SOURCE: "playwright-report/"
+          TARGET: 
"/home/thephpfoundation/preview/web-php-regression-report-pr-${{ 
github.event.number }}/public"
+          SCRIPT_BEFORE: bash /home/thephpfoundation/scripts/pr_created_pre.sh 
web-php-regression-report ${{ github.event.number }}
+
+      - uses: peter-evans/find-comment@v3
+        if: ${{ !cancelled() }}
+        id: snapshot
+        with:
+          issue-number: ${{ github.event.number }}
+          comment-author: 'github-actions[bot]'
+
+      - uses: peter-evans/create-or-update-comment@v4
+        if: ${{ !cancelled() }}
+        with:
+          issue-number: ${{ github.event.number }}
+          comment-id: ${{ steps.snapshot.outputs.comment-id }}
+          edit-mode: 'replace'
+          body: |
+            🚀 Regression report for commit ${{ github.sha }} is at 
https://web-php-regression-report-pr-${{ github.event.number 
}}.preview.thephp.foundation
diff --git a/.github/workflows/update-screenshots.yaml 
b/.github/workflows/update-screenshots.yaml
new file mode 100644
index 0000000000..16e03ccc8f
--- /dev/null
+++ b/.github/workflows/update-screenshots.yaml
@@ -0,0 +1,74 @@
+# https://docs.github.com/en/actions
+
+name: "Update screenshots"
+
+on:
+  workflow_dispatch:
+
+jobs:
+  tests_update_snapshots:
+    name: "Update Tests snapshots"
+
+    runs-on: "ubuntu-latest"
+
+    strategy:
+      matrix:
+        php-version:
+          - "8.2"
+        node-version:
+          - "22.x"
+
+    env:
+      HTTP_HOST: "localhost:8080"
+
+    steps:
+      - name: "Checkout"
+        uses: "actions/checkout@v4"
+
+      - name: "Set up PHP"
+        uses: "shivammathur/setup-php@v2"
+        with:
+          coverage: "none"
+          extensions: "none, curl, dom, json, mbstring, tokenizer, xml, 
xmlwriter"
+          php-version: "${{ matrix.php-version }}"
+
+      - name: Use Node.js ${{ matrix.node-version }}
+        uses: actions/setup-node@v3
+        with:
+          node-version: ${{ matrix.node-version }}
+
+      - name: "Set up problem matchers for PHP"
+        run: "echo \"::add-matcher::${{ runner.tool_cache }}/php.json\""
+
+      - name: "Set up problem matchers for phpunit/phpunit"
+        run: "echo \"::add-matcher::${{ runner.tool_cache }}/phpunit.json\""
+
+      - name: "Determine composer cache directory"
+        run: "echo \"COMPOSER_CACHE_DIR=$(composer config cache-dir)\" >> 
$GITHUB_ENV"
+
+      - name: "Cache dependencies installed with composer"
+        uses: "actions/cache@v3"
+        with:
+          path: "${{ env.COMPOSER_CACHE_DIR }}"
+          key: "php-${{ matrix.php-version }}-composer-${{ 
hashFiles('composer.lock') }}"
+          restore-keys: "php-${{ matrix.php-version }}-composer-"
+
+      - name: "Install dependencies with composer"
+        run: "composer install --ansi --no-interaction --no-progress"
+
+      - name: "Install dependencies"
+        run: "yarn install"
+
+      - name: "Install Playwright"
+        run: "npx playwright install"
+
+      - name: "Run visual tests"
+        run: "make tests_update_snapshots"
+
+      - name: Update snapshots
+        uses: test-room-7/action-update-file@v1
+        if: ${{ !cancelled() }}
+        with:
+          file-path: tests/Visual/**/*
+          commit-msg: Update snapshots
+          github-token: ${{ secrets.GITHUB_TOKEN }}
diff --git a/.gitignore b/.gitignore
index b491def39c..5fa5cd36ef 100644
--- a/.gitignore
+++ b/.gitignore
@@ -5,3 +5,8 @@ backend/mirror.gif
 backend/mirror.png
 backend/mirror.jpg
 backend/GeoIP.dat
+node_modules/
+/test-results/
+/playwright-report/
+/blob-report/
+/playwright/.cache/
diff --git a/.htaccess b/.htaccess
index 912ca12580..918a3f818e 100644
--- a/.htaccess
+++ b/.htaccess
@@ -1,4 +1,4 @@
-<FilesMatch "\.(inc|sql)$">
+<FilesMatch "\.(inc|sql|lock)$">
   Order allow,deny
   Deny from all
 </FilesMatch>
diff --git a/Makefile b/Makefile
index f70a2fe662..2d6e571924 100644
--- a/Makefile
+++ b/Makefile
@@ -1,6 +1,7 @@
 .EXPORT_ALL_VARIABLES:
 
 HTTP_HOST:=localhost:8080
+CORES?=$(shell (nproc  || sysctl -n hw.ncpu) 2> /dev/null)
 
 .PHONY: it
 it: coding-standards tests ## Runs all the targets
@@ -21,7 +22,19 @@ help: ## Displays this list of targets with descriptions
 tests: vendor ## Runs unit and end-to-end tests with phpunit/phpunit
        vendor/bin/phpunit --configuration=tests/phpunit.xml --testsuite=unit
        rm -rf tests/server.log
-       tests/server start; vendor/bin/phpunit 
--configuration=tests/phpunit.xml --testsuite=end-to-end; tests/server stop
+       tests/server start;
+       vendor/bin/phpunit --configuration=tests/phpunit.xml 
--testsuite=end-to-end;
+       tests/server stop
+
+tests_visual:
+       tests/server start;
+       npx playwright test --workers=$(CORES)
+       tests/server stop
+
+tests_update_snapshots:
+       tests/server start;
+       npx playwright test --update-snapshots
+       tests/server stop
 
 vendor: composer.json composer.lock
        composer validate --strict
diff --git a/package.json b/package.json
new file mode 100644
index 0000000000..6678d5ccce
--- /dev/null
+++ b/package.json
@@ -0,0 +1,8 @@
+{
+  "name": "php.net",
+  "devDependencies": {
+    "@playwright/test": "^1.44.0",
+    "@types/node": "^20.12.11"
+  },
+  "scripts": {}
+}
diff --git a/playwright.config.ts b/playwright.config.ts
new file mode 100644
index 0000000000..8951edeead
--- /dev/null
+++ b/playwright.config.ts
@@ -0,0 +1,34 @@
+import {defineConfig, devices} from '@playwright/test';
+
+/**
+ * See https://playwright.dev/docs/test-configuration.
+ */
+export default defineConfig({
+    testDir: './tests/Visual',
+    /* Run tests in files in parallel */
+    fullyParallel: true,
+    /* Fail the build on CI if you accidentally left test.only in the source 
code. */
+    forbidOnly: !!process.env.CI,
+    /* Retry on CI only */
+    retries: process.env.CI ? 2 : 0,
+    /* Opt out of parallel tests on CI. */
+    workers: process.env.CI ? 1 : undefined,
+    /* Reporter to use. See https://playwright.dev/docs/test-reporters */
+    reporter: 'html',
+    /* Shared settings for all the projects below. See 
https://playwright.dev/docs/api/class-testoptions. */
+    use: {
+        /* Base URL to use in actions like `await page.goto('/')`. */
+        // baseURL: 'http://127.0.0.1:3000',
+
+        /* Collect trace when retrying the failed test. See 
https://playwright.dev/docs/trace-viewer */
+        trace: 'on-first-retry',
+    },
+    timeout: 0,
+
+    projects: [
+        {
+            name: 'chromium',
+            use: {...devices['Desktop Chrome']},
+        },
+    ],
+});
diff --git a/tests/Visual/SmokeTest.spec.ts b/tests/Visual/SmokeTest.spec.ts
new file mode 100644
index 0000000000..c475e26d8c
--- /dev/null
+++ b/tests/Visual/SmokeTest.spec.ts
@@ -0,0 +1,63 @@
+import {expect, test} from '@playwright/test';
+
+export type TestPageOptions = {
+    path: string
+    options?: object
+    evaluate?: () => any
+}
+
+const items: TestPageOptions[] = [
+    {
+        path: 'index.php',
+        evaluate: () => {
+            const selector = document.querySelector('.elephpants');
+            selector.remove()
+        }
+    },
+    {
+        path: 'archive/1998.php',
+        evaluate: () => {
+            const selector = document.querySelector('.elephpants');
+            selector.remove()
+        }
+    },
+    {path: 'releases/8_3_6.php'},
+    {path: 'releases/8.0/index.php'},
+    {path: 'releases/8.1/index.php'},
+    {path: 'releases/8.2/index.php'},
+    {path: 'releases/8.3/index.php'},
+    {path: 'manual/index.php'},
+    {path: 'manual/php5.php'},
+    {
+        path: 'conferences/index.php',
+        options: {
+            fullPage: false,
+        }
+    },
+]
+
+for (const item of items) {
+    test(`testing with ${item.path}`, async ({page}, testInfo) => {
+        testInfo.snapshotSuffix = '';
+
+        const httpHost = process.env.HTTP_HOST
+
+        if (typeof httpHost !== 'string') {
+            throw new Error('Environment variable "HTTP_HOST" is not set.')
+        }
+
+        await page.goto(`${httpHost}/${item.path}`)
+
+        if (typeof item.evaluate === 'function') {
+            await page.evaluate(item.evaluate)
+        }
+
+        await expect(page).toHaveScreenshot(
+            `tests/screenshots/${item.path}.png`,
+            item.options ?? {
+                fullPage: true,
+                timeout: 10000,
+            }
+        )
+    })
+}
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-archive-1998-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-archive-1998-php-chromium.png
new file mode 100644
index 0000000000..9148879646
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-archive-1998-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-conferences-index-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-conferences-index-php-chromium.png
new file mode 100644
index 0000000000..87f7701330
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-conferences-index-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-index-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-index-php-chromium.png
new file mode 100644
index 0000000000..6bbd63cf22
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-index-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-index-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-index-php-chromium.png
new file mode 100644
index 0000000000..0a0668de56
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-index-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-php5-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-php5-php-chromium.png
new file mode 100644
index 0000000000..0ba35d00dd
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-manual-php5-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-0-index-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-0-index-php-chromium.png
new file mode 100644
index 0000000000..41c4149248
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-0-index-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-1-index-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-1-index-php-chromium.png
new file mode 100644
index 0000000000..a8fa6a0645
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-1-index-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-2-index-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-2-index-php-chromium.png
new file mode 100644
index 0000000000..fc649a81b9
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-2-index-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-6-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-6-php-chromium.png
new file mode 100644
index 0000000000..438aafe319
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-6-php-chromium.png
 differ
diff --git 
a/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-index-php-chromium.png
 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-index-php-chromium.png
new file mode 100644
index 0000000000..b32afbee0a
Binary files /dev/null and 
b/tests/Visual/SmokeTest.spec.ts-snapshots/tests-screenshots-releases-8-3-index-php-chromium.png
 differ
diff --git a/yarn.lock b/yarn.lock
new file mode 100644
index 0000000000..1f195cd592
--- /dev/null
+++ b/yarn.lock
@@ -0,0 +1,41 @@
+# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+# yarn lockfile v1
+
+
+"@playwright/test@^1.44.0":
+  version "1.44.0"
+  resolved 
"https://registry.yarnpkg.com/@playwright/test/-/test-1.44.0.tgz#ac7a764b5ee6a80558bdc0fcbc525fcb81f83465";
+  integrity 
sha512-rNX5lbNidamSUorBhB4XZ9SQTjAqfe5M+p37Z8ic0jPFBMo5iCtQz1kRWkEMg+rYOKSlVycpQmpqjSFq7LXOfg==
+  dependencies:
+    playwright "1.44.0"
+
+"@types/node@^20.12.11":
+  version "20.12.11"
+  resolved 
"https://registry.yarnpkg.com/@types/node/-/node-20.12.11.tgz#c4ef00d3507000d17690643278a60dc55a9dc9be";
+  integrity 
sha512-vDg9PZ/zi+Nqp6boSOT7plNuthRugEKixDv5sFTIpkE89MmNtEArAShI4mxuX2+UrLEe9pxC1vm2cjm9YlWbJw==
+  dependencies:
+    undici-types "~5.26.4"
+
+fsevents@2.3.2:
+  version "2.3.2"
+  resolved 
"https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a";
+  integrity 
sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==
+
+playwright-core@1.44.0:
+  version "1.44.0"
+  resolved 
"https://registry.yarnpkg.com/playwright-core/-/playwright-core-1.44.0.tgz#316c4f0bca0551ffb88b6eb1c97bc0d2d861b0d5";
+  integrity 
sha512-ZTbkNpFfYcGWohvTTl+xewITm7EOuqIqex0c7dNZ+aXsbrLj0qI8XlGKfPpipjm0Wny/4Lt4CJsWJk1stVS5qQ==
+
+playwright@1.44.0:
+  version "1.44.0"
+  resolved 
"https://registry.yarnpkg.com/playwright/-/playwright-1.44.0.tgz#22894e9b69087f6beb639249323d80fe2b5087ff";
+  integrity 
sha512-F9b3GUCLQ3Nffrfb6dunPOkE5Mh68tR7zN32L4jCk4FjQamgesGay7/dAAe1WaMEGV04DkdJfcJzjoCKygUaRQ==
+  dependencies:
+    playwright-core "1.44.0"
+  optionalDependencies:
+    fsevents "2.3.2"
+
+undici-types@~5.26.4:
+  version "5.26.5"
+  resolved 
"https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617";
+  integrity 
sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==

Reply via email to