mirror of
https://github.com/pestphp/pest.git
synced 2026-06-05 10:52:14 +02:00
Compare commits
196 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 92e76eb5ab | |||
| bd22f478b8 | |||
| eeaac34cf6 | |||
| b9b07d8983 | |||
| 6aa7d2f891 | |||
| 1c21a7647a | |||
| d649de1988 | |||
| 783ca4bcd6 | |||
| ba07497219 | |||
| 1ca021dea6 | |||
| 2fc75cfcf0 | |||
| 6cc48f63f8 | |||
| e0419d1328 | |||
| faa6988801 | |||
| c12247fafd | |||
| 29b4452443 | |||
| 1b168aba1c | |||
| 6aabd977cd | |||
| a882543c53 | |||
| c250b9da4f | |||
| 46bc3dc628 | |||
| d3ce498b8a | |||
| e1a4b98b71 | |||
| 9afbcd5c18 | |||
| 75593b6454 | |||
| 89590d6120 | |||
| fb0978c9bf | |||
| a3796daa42 | |||
| e3004db666 | |||
| 99cc4e0146 | |||
| a47e6f8fef | |||
| 536d79f765 | |||
| 65c0fbc528 | |||
| 9e4cf4b665 | |||
| 7bea819978 | |||
| 4280233b40 | |||
| d6db3a8a20 | |||
| 51c8ce4df6 | |||
| 5b8393b925 | |||
| e4d9b61fdf | |||
| e2d940cd53 | |||
| 380ccd30b4 | |||
| 31c200716d | |||
| 6add4da543 | |||
| 8ddcd3e853 | |||
| e3e178fd94 | |||
| 7b1ec9f003 | |||
| 1e48c5d473 | |||
| d00ec95dd9 | |||
| 89f3d6cb39 | |||
| a07a2e512a | |||
| 57eecb2b3d | |||
| 9f804dc954 | |||
| 7cbad4c589 | |||
| 5cae93b059 | |||
| df829ad19d | |||
| 635460653c | |||
| 1aa80dc398 | |||
| 8a14056111 | |||
| f247dd8e7b | |||
| 1c7c9754fd | |||
| 5f37939fda | |||
| 28305fcb7a | |||
| 5242803694 | |||
| 925935a7e8 | |||
| 460401c379 | |||
| 348b439172 | |||
| a4e77766c5 | |||
| 4a8c2d7d78 | |||
| 7d51601120 | |||
| 631bbe318b | |||
| 9b7c15d5b6 | |||
| 872796bd9b | |||
| c38d32ae86 | |||
| 6407c4f78f | |||
| 6e1bf63f6a | |||
| 1d3e8bb5dd | |||
| 3cc9b169e3 | |||
| c4911d046b | |||
| d0295f6168 | |||
| 21efbc3107 | |||
| e59b99cd73 | |||
| bf48e20880 | |||
| 53db68e005 | |||
| 34f1e9a7f2 | |||
| 57fd5ce042 | |||
| 3bcabfb63b | |||
| aa3a7c303a | |||
| 5c08a135f7 | |||
| 6e0e030d71 | |||
| b2c07561e7 | |||
| 97600b6f0b | |||
| 8a51f15d65 | |||
| a349f53964 | |||
| a725e774c0 | |||
| bed5e5b54a | |||
| 45b1d4ce20 | |||
| d106b70766 | |||
| 6ac6c1518e | |||
| fda515a17f | |||
| 0a97d3a288 | |||
| 3802fa80e6 | |||
| 5c3cbc14d2 | |||
| 6b9c768172 | |||
| 4a2fc179ae | |||
| b5bb2139dc | |||
| 07416a3c61 | |||
| 30b94e3034 | |||
| be34eecb2f | |||
| 5d9f95f8d4 | |||
| 48b70a03d5 | |||
| 4b8642b972 | |||
| 8711d51eac | |||
| 58dfb6da64 | |||
| d7735d1faa | |||
| 6b59166f3c | |||
| 3a26028d17 | |||
| 3c91bf4ad2 | |||
| 6a434be0f6 | |||
| f355b99bbf | |||
| 95a00341e9 | |||
| 466259646d | |||
| 00f8d56083 | |||
| ca2dca592d | |||
| 405d8d4406 | |||
| b944ee5841 | |||
| f4e22dcafe | |||
| 339c1e8cac | |||
| d4c7362132 | |||
| 81bfdbf8fe | |||
| f45cbf43c5 | |||
| b9088d23fb | |||
| 7250185423 | |||
| e457eb0e9c | |||
| 48357c6f30 | |||
| b46f051550 | |||
| 3d3c5d41ac | |||
| caabebf2a1 | |||
| 470a5833d4 | |||
| c1feefbb9e | |||
| e876dba8ba | |||
| d9c18f9c02 | |||
| 660b57b365 | |||
| 68527c996f | |||
| c6a42a2b28 | |||
| fcf5c27914 | |||
| 856a370032 | |||
| e24882c486 | |||
| 51fc380789 | |||
| f6609f4039 | |||
| 2941f9821f | |||
| ed399af43e | |||
| 0d66dc4322 | |||
| 7e4280bf83 | |||
| a5915b16ab | |||
| 1476b529a1 | |||
| 2892341c28 | |||
| 59e781e77b | |||
| 55a3394f8c | |||
| 0d99c33c4e | |||
| adc5aae6f8 | |||
| 980667e845 | |||
| 8c849c5f40 | |||
| 47f1fc2d94 | |||
| 9c8033d60c | |||
| 42d1092a9e | |||
| c7e32f5d33 | |||
| d379128cc4 | |||
| f09d6f2064 | |||
| 494cc6e2a4 | |||
| f52a455773 | |||
| 184f5d2742 | |||
| 1d81069a2a | |||
| 4b9bb77b54 | |||
| c440031e28 | |||
| bff44562a9 | |||
| 9ebb990f96 | |||
| cabff738f7 | |||
| 0746173a32 | |||
| 87db0b4847 | |||
| 6ba373a772 | |||
| 945d476409 | |||
| a8cf0fe2cb | |||
| 2ae072bb95 | |||
| 59d066950c | |||
| 0dd1aa72ef | |||
| 4e03cd3edb | |||
| eeab24e2bb | |||
| 9b64d5425a | |||
| 0acab1cbb4 | |||
| e616eab9fb | |||
| 7cbb1fcdb2 | |||
| cb5f6e1bd2 | |||
| 985dadd934 | |||
| 10aee6045c | |||
| 4ac14b2528 |
13
.github/SECURITY.md
vendored
Normal file
13
.github/SECURITY.md
vendored
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Security Policy
|
||||||
|
|
||||||
|
**PLEASE DON'T DISCLOSE SECURITY-RELATED ISSUES PUBLICLY, [SEE BELOW](#reporting-a-vulnerability).**
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
If you discover a security vulnerability in Pest, please report it privately using one of the following channels:
|
||||||
|
|
||||||
|
1. **GitHub Private Vulnerability Reporting** (preferred) — go to the repository's **Security** tab and click **"Report a vulnerability"**. This creates a private advisory visible only to maintainers and provides a structured workflow for triage, fix coordination, and CVE assignment.
|
||||||
|
|
||||||
|
2. **Email** — send the details to Nuno Maduro at **enunomaduro@gmail.com**.
|
||||||
|
|
||||||
|
All security vulnerabilities will be promptly addressed.
|
||||||
19
.github/dependabot.yml
vendored
Normal file
19
.github/dependabot.yml
vendored
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "weekly"
|
||||||
|
groups:
|
||||||
|
github-actions:
|
||||||
|
patterns:
|
||||||
|
- "*"
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "weekly"
|
||||||
|
target-branch: "5.x"
|
||||||
|
groups:
|
||||||
|
github-actions:
|
||||||
|
patterns:
|
||||||
|
- "*"
|
||||||
13
.github/workflows/static.yml
vendored
13
.github/workflows/static.yml
vendored
@ -11,6 +11,9 @@ concurrency:
|
|||||||
group: static-${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
group: static-${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
||||||
cancel-in-progress: true
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
static:
|
static:
|
||||||
if: github.event_name != 'schedule' || github.repository == 'pestphp/pest'
|
if: github.event_name != 'schedule' || github.repository == 'pestphp/pest'
|
||||||
@ -21,14 +24,14 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
fail-fast: true
|
fail-fast: true
|
||||||
matrix:
|
matrix:
|
||||||
dependency-version: [prefer-lowest, prefer-stable]
|
dependency-version: [prefer-stable]
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v6
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||||
|
|
||||||
- name: Setup PHP
|
- name: Setup PHP
|
||||||
uses: shivammathur/setup-php@v2
|
uses: shivammathur/setup-php@accd6127cb78bee3e8082180cb391013d204ef9f # v2
|
||||||
with:
|
with:
|
||||||
php-version: 8.3
|
php-version: 8.3
|
||||||
tools: composer:v2
|
tools: composer:v2
|
||||||
@ -41,10 +44,10 @@ jobs:
|
|||||||
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Cache Composer dependencies
|
- name: Cache Composer dependencies
|
||||||
uses: actions/cache@v5
|
uses: actions/cache@27d5ce7f107fe9357f9df03efb73ab90386fccae # v5
|
||||||
with:
|
with:
|
||||||
path: ${{ steps.composer-cache.outputs.dir }}
|
path: ${{ steps.composer-cache.outputs.dir }}
|
||||||
key: static-php-8.3-${{ matrix.dependency-version }}-composer-${{ hashFiles('**/composer.json') }}
|
key: static-php-8.3-${{ matrix.dependency-version }}-composer-${{ hashFiles('**/composer.json', '**/composer.lock') }}
|
||||||
restore-keys: |
|
restore-keys: |
|
||||||
static-php-8.3-${{ matrix.dependency-version }}-composer-
|
static-php-8.3-${{ matrix.dependency-version }}-composer-
|
||||||
static-php-8.3-composer-
|
static-php-8.3-composer-
|
||||||
|
|||||||
11
.github/workflows/tests.yml
vendored
11
.github/workflows/tests.yml
vendored
@ -11,6 +11,9 @@ concurrency:
|
|||||||
group: tests-${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
group: tests-${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
||||||
cancel-in-progress: true
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
tests:
|
tests:
|
||||||
if: github.event_name != 'schedule' || github.repository == 'pestphp/pest'
|
if: github.event_name != 'schedule' || github.repository == 'pestphp/pest'
|
||||||
@ -32,10 +35,10 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v6
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||||
|
|
||||||
- name: Setup PHP
|
- name: Setup PHP
|
||||||
uses: shivammathur/setup-php@v2
|
uses: shivammathur/setup-php@accd6127cb78bee3e8082180cb391013d204ef9f # v2
|
||||||
with:
|
with:
|
||||||
php-version: ${{ matrix.php }}
|
php-version: ${{ matrix.php }}
|
||||||
tools: composer:v2
|
tools: composer:v2
|
||||||
@ -48,10 +51,10 @@ jobs:
|
|||||||
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Cache Composer dependencies
|
- name: Cache Composer dependencies
|
||||||
uses: actions/cache@v5
|
uses: actions/cache@27d5ce7f107fe9357f9df03efb73ab90386fccae # v5
|
||||||
with:
|
with:
|
||||||
path: ${{ steps.composer-cache.outputs.dir }}
|
path: ${{ steps.composer-cache.outputs.dir }}
|
||||||
key: ${{ matrix.os }}-php-${{ matrix.php }}-symfony-${{ matrix.symfony }}-composer-${{ hashFiles('**/composer.json') }}
|
key: ${{ matrix.os }}-php-${{ matrix.php }}-symfony-${{ matrix.symfony }}-composer-${{ hashFiles('**/composer.json', '**/composer.lock') }}
|
||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ matrix.os }}-php-${{ matrix.php }}-symfony-${{ matrix.symfony }}-composer-
|
${{ matrix.os }}-php-${{ matrix.php }}-symfony-${{ matrix.symfony }}-composer-
|
||||||
${{ matrix.os }}-php-${{ matrix.php }}-composer-
|
${{ matrix.os }}-php-${{ matrix.php }}-composer-
|
||||||
|
|||||||
@ -6,6 +6,7 @@
|
|||||||
<a href="https://packagist.org/packages/pestphp/pest"><img alt="Latest Version" src="https://img.shields.io/packagist/v/pestphp/pest"></a>
|
<a href="https://packagist.org/packages/pestphp/pest"><img alt="Latest Version" src="https://img.shields.io/packagist/v/pestphp/pest"></a>
|
||||||
<a href="https://packagist.org/packages/pestphp/pest"><img alt="License" src="https://img.shields.io/packagist/l/pestphp/pest"></a>
|
<a href="https://packagist.org/packages/pestphp/pest"><img alt="License" src="https://img.shields.io/packagist/l/pestphp/pest"></a>
|
||||||
<a href="https://whyphp.dev"><img src="https://img.shields.io/badge/Why_PHP-in_2026-7A86E8?style=flat-square&labelColor=18181b" alt="Why PHP in 2026"></a>
|
<a href="https://whyphp.dev"><img src="https://img.shields.io/badge/Why_PHP-in_2026-7A86E8?style=flat-square&labelColor=18181b" alt="Why PHP in 2026"></a>
|
||||||
|
<a href="https://youtube.com/@nunomaduro?sub_confirmation=1"><img alt="YouTube Channel Subscribers" src="https://img.shields.io/youtube/channel/subscribers/UCO_hYZF2gb_CyG5sA7ArlGg?style=flat&label=youtube&color=brightgreen"></a>
|
||||||
</p>
|
</p>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
|||||||
12
bin/pest
12
bin/pest
@ -3,8 +3,10 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
use Pest\Contracts\Restarter;
|
||||||
use Pest\Kernel;
|
use Pest\Kernel;
|
||||||
use Pest\Panic;
|
use Pest\Panic;
|
||||||
|
use Pest\Support\Container;
|
||||||
use Pest\TestCaseFilters\GitDirtyTestCaseFilter;
|
use Pest\TestCaseFilters\GitDirtyTestCaseFilter;
|
||||||
use Pest\TestCaseMethodFilters\AssigneeTestCaseFilter;
|
use Pest\TestCaseMethodFilters\AssigneeTestCaseFilter;
|
||||||
use Pest\TestCaseMethodFilters\IssueTestCaseFilter;
|
use Pest\TestCaseMethodFilters\IssueTestCaseFilter;
|
||||||
@ -142,6 +144,7 @@ use Symfony\Component\Console\Output\ConsoleOutput;
|
|||||||
|
|
||||||
// Get $rootPath based on $autoloadPath
|
// Get $rootPath based on $autoloadPath
|
||||||
$rootPath = dirname($autoloadPath, 2);
|
$rootPath = dirname($autoloadPath, 2);
|
||||||
|
|
||||||
$input = new ArgvInput;
|
$input = new ArgvInput;
|
||||||
|
|
||||||
$testSuite = TestSuite::getInstance(
|
$testSuite = TestSuite::getInstance(
|
||||||
@ -192,6 +195,15 @@ use Symfony\Component\Console\Output\ConsoleOutput;
|
|||||||
try {
|
try {
|
||||||
$kernel = Kernel::boot($testSuite, $input, $output);
|
$kernel = Kernel::boot($testSuite, $input, $output);
|
||||||
|
|
||||||
|
$container = Container::getInstance();
|
||||||
|
|
||||||
|
foreach (Kernel::RESTARTERS as $restarterClass) {
|
||||||
|
$restarter = $container->get($restarterClass);
|
||||||
|
assert($restarter instanceof Restarter);
|
||||||
|
|
||||||
|
$restarter->maybeRestart($rootPath, $originalArguments);
|
||||||
|
}
|
||||||
|
|
||||||
$result = $kernel->handle($originalArguments, $arguments);
|
$result = $kernel->handle($originalArguments, $arguments);
|
||||||
|
|
||||||
$kernel->terminate();
|
$kernel->terminate();
|
||||||
|
|||||||
239
bin/pest-tia-vite-deps.mjs
Normal file
239
bin/pest-tia-vite-deps.mjs
Normal file
@ -0,0 +1,239 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
import { readdir, readFile } from 'node:fs/promises'
|
||||||
|
import { existsSync } from 'node:fs'
|
||||||
|
import { createRequire } from 'node:module'
|
||||||
|
import { resolve, relative, extname, sep, join } from 'node:path'
|
||||||
|
import { pathToFileURL } from 'node:url'
|
||||||
|
|
||||||
|
const PAGE_EXTENSIONS = new Set([
|
||||||
|
'.vue', '.svelte',
|
||||||
|
'.tsx', '.jsx',
|
||||||
|
'.ts', '.js',
|
||||||
|
'.mts', '.cts', '.mjs', '.cjs',
|
||||||
|
])
|
||||||
|
const ASSET_EXT_RE = /\.(css|scss|sass|less|styl|stylus|svg|png|jpe?g|gif|webp|avif|ico|bmp|woff2?|ttf|eot|otf|md|mdx|txt|html|mp4|webm|mp3|wav|ogg|m4a|pdf|wasm|glsl|frag|vert)$/i
|
||||||
|
const PROJECT_ROOT = resolve(process.argv[2] ?? process.cwd())
|
||||||
|
const PAGE_DIR_CANDIDATES = [
|
||||||
|
'resources/js/Pages',
|
||||||
|
'resources/js/pages',
|
||||||
|
'assets/js/Pages',
|
||||||
|
'assets/js/pages',
|
||||||
|
'assets/Pages',
|
||||||
|
'assets/pages',
|
||||||
|
]
|
||||||
|
|
||||||
|
async function loadRolldown() {
|
||||||
|
const projectRequire = createRequire(join(PROJECT_ROOT, 'package.json'))
|
||||||
|
const path = projectRequire.resolve('rolldown')
|
||||||
|
return await import(pathToFileURL(path).href)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function readJsonWithComments(path) {
|
||||||
|
const raw = await readFile(path, 'utf8')
|
||||||
|
const stripped = raw
|
||||||
|
.replace(/\/\*[\s\S]*?\*\//g, '')
|
||||||
|
.replace(/(^|[^:])\/\/[^\n]*/g, '$1')
|
||||||
|
return JSON.parse(stripped)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadAliasFromTsconfig() {
|
||||||
|
const alias = {}
|
||||||
|
for (const name of ['tsconfig.json', 'jsconfig.json']) {
|
||||||
|
const p = join(PROJECT_ROOT, name)
|
||||||
|
if (!existsSync(p)) continue
|
||||||
|
let cfg
|
||||||
|
try { cfg = await readJsonWithComments(p) } catch { continue }
|
||||||
|
const baseUrl = resolve(PROJECT_ROOT, cfg?.compilerOptions?.baseUrl ?? '.')
|
||||||
|
const paths = cfg?.compilerOptions?.paths ?? {}
|
||||||
|
for (const [key, targets] of Object.entries(paths)) {
|
||||||
|
if (!key.endsWith('/*')) continue
|
||||||
|
const t0 = Array.isArray(targets) ? targets[0] : null
|
||||||
|
if (typeof t0 !== 'string' || !t0.endsWith('/*')) continue
|
||||||
|
const aliasKey = key.slice(0, -2)
|
||||||
|
if (alias[aliasKey] !== undefined) continue
|
||||||
|
alias[aliasKey] = resolve(baseUrl, t0.slice(0, -2))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return alias
|
||||||
|
}
|
||||||
|
|
||||||
|
async function listPageFiles(pagesDir) {
|
||||||
|
if (!existsSync(pagesDir)) return []
|
||||||
|
|
||||||
|
const out = []
|
||||||
|
const walk = async (dir) => {
|
||||||
|
let entries
|
||||||
|
try { entries = await readdir(dir, { withFileTypes: true }) } catch { return }
|
||||||
|
for (const entry of entries) {
|
||||||
|
const full = resolve(dir, entry.name)
|
||||||
|
if (entry.isDirectory()) { await walk(full); continue }
|
||||||
|
if (PAGE_EXTENSIONS.has(extname(entry.name))) out.push(full)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await walk(pagesDir)
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
async function discoverPagesDir() {
|
||||||
|
const override = process.env.TIA_VITE_PAGES_DIR
|
||||||
|
if (override && override.length > 0) {
|
||||||
|
return resolve(PROJECT_ROOT, override.replace(/\\/g, '/'))
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const rel of PAGE_DIR_CANDIDATES) {
|
||||||
|
const abs = resolve(PROJECT_ROOT, rel)
|
||||||
|
if (!existsSync(abs)) continue
|
||||||
|
const files = await listPageFiles(abs)
|
||||||
|
if (files.length > 0) return abs
|
||||||
|
}
|
||||||
|
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
function componentNameFor(pageAbs, pagesDir) {
|
||||||
|
const rel = relative(pagesDir, pageAbs).split(sep).join('/')
|
||||||
|
const ext = extname(rel)
|
||||||
|
return rel.slice(0, rel.length - ext.length)
|
||||||
|
}
|
||||||
|
|
||||||
|
function isLocalSpecifier(source, aliasKeys) {
|
||||||
|
if (source.startsWith('.') || source.startsWith('/')) return true
|
||||||
|
for (const key of aliasKeys) {
|
||||||
|
if (source === key || source.startsWith(key + '/')) return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
const pagesDir = await discoverPagesDir()
|
||||||
|
|
||||||
|
if (pagesDir === null) {
|
||||||
|
process.stdout.write('{}')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const pages = await listPageFiles(pagesDir)
|
||||||
|
|
||||||
|
if (pages.length === 0) {
|
||||||
|
process.stdout.write('{}')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const { rolldown } = await loadRolldown()
|
||||||
|
const alias = await loadAliasFromTsconfig()
|
||||||
|
const aliasKeys = Object.keys(alias)
|
||||||
|
|
||||||
|
const graph = new Map()
|
||||||
|
|
||||||
|
const collector = {
|
||||||
|
name: 'pest-tia-collector',
|
||||||
|
moduleParsed(info) {
|
||||||
|
const id = info.id
|
||||||
|
if (!id || id.startsWith('\0')) return
|
||||||
|
const deps = new Set()
|
||||||
|
for (const i of info.importedIds) if (i && !i.startsWith('\0')) deps.add(i)
|
||||||
|
for (const i of info.dynamicallyImportedIds) if (i && !i.startsWith('\0')) deps.add(i)
|
||||||
|
graph.set(id, deps)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const externalBare = {
|
||||||
|
name: 'pest-tia-external-bare',
|
||||||
|
resolveId(source) {
|
||||||
|
if (!source) return null
|
||||||
|
if (isLocalSpecifier(source, aliasKeys)) return null
|
||||||
|
return { id: source, external: true }
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const assetStub = {
|
||||||
|
name: 'pest-tia-asset-stub',
|
||||||
|
load(id) {
|
||||||
|
if (!id) return null
|
||||||
|
if (ASSET_EXT_RE.test(id)) {
|
||||||
|
return { code: 'export default null', moduleSideEffects: false }
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const input = Object.create(null)
|
||||||
|
for (let i = 0; i < pages.length; i++) input[`p${i}`] = pages[i]
|
||||||
|
|
||||||
|
const bundle = await rolldown({
|
||||||
|
input,
|
||||||
|
cwd: PROJECT_ROOT,
|
||||||
|
resolve: {
|
||||||
|
alias,
|
||||||
|
extensions: ['.tsx', '.ts', '.jsx', '.js', '.mjs', '.cjs', '.json'],
|
||||||
|
},
|
||||||
|
transform: { jsx: 'preserve' },
|
||||||
|
treeshake: false,
|
||||||
|
plugins: [externalBare, assetStub, collector],
|
||||||
|
logLevel: 'silent',
|
||||||
|
onLog: () => {},
|
||||||
|
})
|
||||||
|
|
||||||
|
try {
|
||||||
|
await bundle.generate({ format: 'esm' })
|
||||||
|
} finally {
|
||||||
|
await bundle.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
const reverse = new Map()
|
||||||
|
const transitiveCache = new Map()
|
||||||
|
|
||||||
|
const computeTransitive = (id, stack) => {
|
||||||
|
const cached = transitiveCache.get(id)
|
||||||
|
if (cached) return cached
|
||||||
|
if (stack.has(id)) return null
|
||||||
|
|
||||||
|
stack.add(id)
|
||||||
|
const acc = new Set()
|
||||||
|
const deps = graph.get(id)
|
||||||
|
if (deps) {
|
||||||
|
for (const dep of deps) {
|
||||||
|
if (!dep || dep.startsWith('\0')) continue
|
||||||
|
if (dep.startsWith(PROJECT_ROOT)) {
|
||||||
|
const rel = relative(PROJECT_ROOT, dep).split(sep).join('/')
|
||||||
|
acc.add(rel)
|
||||||
|
}
|
||||||
|
if (stack.has(dep)) continue
|
||||||
|
const child = computeTransitive(dep, stack)
|
||||||
|
if (child) for (const r of child) acc.add(r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
stack.delete(id)
|
||||||
|
transitiveCache.set(id, acc)
|
||||||
|
return acc
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const page of pages) {
|
||||||
|
const pageComponent = componentNameFor(page, pagesDir)
|
||||||
|
const reachable = computeTransitive(page, new Set())
|
||||||
|
if (!reachable) continue
|
||||||
|
for (const rel of reachable) {
|
||||||
|
const bucket = reverse.get(rel) ?? new Set()
|
||||||
|
bucket.add(pageComponent)
|
||||||
|
reverse.set(rel, bucket)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = Object.create(null)
|
||||||
|
const keys = [...reverse.keys()].sort()
|
||||||
|
for (const key of keys) {
|
||||||
|
payload[key] = [...reverse.get(key)].sort()
|
||||||
|
}
|
||||||
|
|
||||||
|
process.stdout.write(JSON.stringify(payload))
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
void pathToFileURL
|
||||||
|
await main()
|
||||||
|
} catch (err) {
|
||||||
|
process.stderr.write(String(err?.stack ?? err ?? 'unknown error'))
|
||||||
|
process.exit(1)
|
||||||
|
}
|
||||||
@ -6,6 +6,7 @@ use ParaTest\WrapperRunner\ApplicationForWrapperWorker;
|
|||||||
use ParaTest\WrapperRunner\WrapperWorker;
|
use ParaTest\WrapperRunner\WrapperWorker;
|
||||||
use Pest\Kernel;
|
use Pest\Kernel;
|
||||||
use Pest\Plugins\Actions\CallsHandleArguments;
|
use Pest\Plugins\Actions\CallsHandleArguments;
|
||||||
|
use Pest\Support\Container;
|
||||||
use Pest\TestSuite;
|
use Pest\TestSuite;
|
||||||
use Symfony\Component\Console\Input\ArgvInput;
|
use Symfony\Component\Console\Input\ArgvInput;
|
||||||
use Symfony\Component\Console\Output\ConsoleOutput;
|
use Symfony\Component\Console\Output\ConsoleOutput;
|
||||||
@ -58,6 +59,15 @@ $bootPest = (static function (): void {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$container = Container::getInstance();
|
||||||
|
$rootPath = dirname(PHPUNIT_COMPOSER_INSTALL, 2);
|
||||||
|
|
||||||
|
foreach (Kernel::RESTARTERS as $restarterClass) {
|
||||||
|
$restarter = $container->get($restarterClass);
|
||||||
|
|
||||||
|
$restarter->maybeRestart($rootPath, $_SERVER['argv']);
|
||||||
|
}
|
||||||
|
|
||||||
assert(isset($getopt['status-file']) && is_string($getopt['status-file']));
|
assert(isset($getopt['status-file']) && is_string($getopt['status-file']));
|
||||||
$statusFile = fopen($getopt['status-file'], 'wb');
|
$statusFile = fopen($getopt['status-file'], 'wb');
|
||||||
assert(is_resource($statusFile));
|
assert(is_resource($statusFile));
|
||||||
|
|||||||
@ -19,18 +19,19 @@
|
|||||||
"require": {
|
"require": {
|
||||||
"php": "^8.3.0",
|
"php": "^8.3.0",
|
||||||
"brianium/paratest": "^7.20.0",
|
"brianium/paratest": "^7.20.0",
|
||||||
"nunomaduro/collision": "^8.9.3",
|
"composer/xdebug-handler": "^3.0.5",
|
||||||
|
"nunomaduro/collision": "^8.9.4",
|
||||||
"nunomaduro/termwind": "^2.4.0",
|
"nunomaduro/termwind": "^2.4.0",
|
||||||
"pestphp/pest-plugin": "^4.0.0",
|
"pestphp/pest-plugin": "^4.0.0",
|
||||||
"pestphp/pest-plugin-arch": "^4.0.2",
|
"pestphp/pest-plugin-arch": "^4.0.2",
|
||||||
"pestphp/pest-plugin-mutate": "^4.0.1",
|
"pestphp/pest-plugin-mutate": "^4.0.1",
|
||||||
"pestphp/pest-plugin-profanity": "^4.2.1",
|
"pestphp/pest-plugin-profanity": "^4.2.1",
|
||||||
"phpunit/phpunit": "^12.5.16",
|
"phpunit/phpunit": "^12.5.28",
|
||||||
"symfony/process": "^7.4.8|^8.0.8"
|
"symfony/process": "^7.4.13|^8.1.0"
|
||||||
},
|
},
|
||||||
"conflict": {
|
"conflict": {
|
||||||
"filp/whoops": "<2.18.3",
|
"filp/whoops": "<2.18.3",
|
||||||
"phpunit/phpunit": ">12.5.16",
|
"phpunit/phpunit": ">12.5.28",
|
||||||
"sebastian/exporter": "<7.0.0",
|
"sebastian/exporter": "<7.0.0",
|
||||||
"webmozart/assert": "<1.11.0"
|
"webmozart/assert": "<1.11.0"
|
||||||
},
|
},
|
||||||
@ -58,11 +59,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"mrpunyapal/peststan": "^0.2.5",
|
"mrpunyapal/peststan": "^0.2.10",
|
||||||
"pestphp/pest-dev-tools": "^4.1.0",
|
"pestphp/pest-dev-tools": "^4.1.0",
|
||||||
"pestphp/pest-plugin-browser": "^4.3.1",
|
"pestphp/pest-plugin-browser": "^4.3.1",
|
||||||
"pestphp/pest-plugin-type-coverage": "^4.0.4",
|
"pestphp/pest-plugin-type-coverage": "^4.0.4",
|
||||||
"psy/psysh": "^0.12.22"
|
"psy/psysh": "^0.12.23"
|
||||||
},
|
},
|
||||||
"minimum-stability": "dev",
|
"minimum-stability": "dev",
|
||||||
"prefer-stable": true,
|
"prefer-stable": true,
|
||||||
@ -123,6 +124,7 @@
|
|||||||
"Pest\\Plugins\\Verbose",
|
"Pest\\Plugins\\Verbose",
|
||||||
"Pest\\Plugins\\Version",
|
"Pest\\Plugins\\Version",
|
||||||
"Pest\\Plugins\\Shard",
|
"Pest\\Plugins\\Shard",
|
||||||
|
"Pest\\Plugins\\Tia",
|
||||||
"Pest\\Plugins\\Parallel"
|
"Pest\\Plugins\\Parallel"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|||||||
@ -1,6 +1,39 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
* BSD 3-Clause License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2001-2023, Sebastian Bergmann
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* 3. Neither the name of the copyright holder nor the names of its
|
||||||
|
* contributors may be used to endorse or promote products derived from
|
||||||
|
* this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This file is part of PHPUnit.
|
* This file is part of PHPUnit.
|
||||||
*
|
*
|
||||||
|
|||||||
388
overrides/Runner/TestSuiteSorter.php
Normal file
388
overrides/Runner/TestSuiteSorter.php
Normal file
@ -0,0 +1,388 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
* BSD 3-Clause License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2001-2023, Sebastian Bergmann
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* 3. Neither the name of the copyright holder nor the names of its
|
||||||
|
* contributors may be used to endorse or promote products derived from
|
||||||
|
* this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This file is part of PHPUnit.
|
||||||
|
*
|
||||||
|
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||||
|
*
|
||||||
|
* For the full copyright and license information, please view the LICENSE
|
||||||
|
* file that was distributed with this source code.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace PHPUnit\Runner;
|
||||||
|
|
||||||
|
use PHPUnit\Framework\DataProviderTestSuite;
|
||||||
|
use PHPUnit\Framework\Reorderable;
|
||||||
|
use PHPUnit\Framework\Test;
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
use PHPUnit\Framework\TestSuite;
|
||||||
|
use PHPUnit\Runner\ResultCache\NullResultCache;
|
||||||
|
use PHPUnit\Runner\ResultCache\ResultCache;
|
||||||
|
use PHPUnit\Runner\ResultCache\ResultCacheId;
|
||||||
|
|
||||||
|
use function array_diff;
|
||||||
|
use function array_merge;
|
||||||
|
use function array_reverse;
|
||||||
|
use function array_splice;
|
||||||
|
use function assert;
|
||||||
|
use function count;
|
||||||
|
use function in_array;
|
||||||
|
use function max;
|
||||||
|
use function shuffle;
|
||||||
|
use function usort;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal This class is not covered by the backward compatibility promise for PHPUnit
|
||||||
|
*/
|
||||||
|
final class TestSuiteSorter
|
||||||
|
{
|
||||||
|
public const int ORDER_DEFAULT = 0;
|
||||||
|
|
||||||
|
public const int ORDER_RANDOMIZED = 1;
|
||||||
|
|
||||||
|
public const int ORDER_REVERSED = 2;
|
||||||
|
|
||||||
|
public const int ORDER_DEFECTS_FIRST = 3;
|
||||||
|
|
||||||
|
public const int ORDER_DURATION = 4;
|
||||||
|
|
||||||
|
public const int ORDER_SIZE = 5;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var non-empty-array<non-empty-string, positive-int>
|
||||||
|
*/
|
||||||
|
private const array SIZE_SORT_WEIGHT = [
|
||||||
|
'small' => 1,
|
||||||
|
'medium' => 2,
|
||||||
|
'large' => 3,
|
||||||
|
'unknown' => 4,
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<string, int> Associative array of (string => DEFECT_SORT_WEIGHT) elements
|
||||||
|
*/
|
||||||
|
private array $defectSortOrder = [];
|
||||||
|
|
||||||
|
private readonly ResultCache $cache;
|
||||||
|
|
||||||
|
public function __construct(?ResultCache $cache = null)
|
||||||
|
{
|
||||||
|
$this->cache = $cache ?? new NullResultCache;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function reorderTestsInSuite(Test $suite, int $order, bool $resolveDependencies, int $orderDefects): void
|
||||||
|
{
|
||||||
|
$allowedOrders = [
|
||||||
|
self::ORDER_DEFAULT,
|
||||||
|
self::ORDER_REVERSED,
|
||||||
|
self::ORDER_RANDOMIZED,
|
||||||
|
self::ORDER_DURATION,
|
||||||
|
self::ORDER_SIZE,
|
||||||
|
];
|
||||||
|
|
||||||
|
if (! in_array($order, $allowedOrders, true)) {
|
||||||
|
// @codeCoverageIgnoreStart
|
||||||
|
throw new InvalidOrderException;
|
||||||
|
// @codeCoverageIgnoreEnd
|
||||||
|
}
|
||||||
|
|
||||||
|
$allowedOrderDefects = [
|
||||||
|
self::ORDER_DEFAULT,
|
||||||
|
self::ORDER_DEFECTS_FIRST,
|
||||||
|
];
|
||||||
|
|
||||||
|
if (! in_array($orderDefects, $allowedOrderDefects, true)) {
|
||||||
|
// @codeCoverageIgnoreStart
|
||||||
|
throw new InvalidOrderException;
|
||||||
|
// @codeCoverageIgnoreEnd
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($suite instanceof TestSuite) {
|
||||||
|
foreach ($suite as $_suite) {
|
||||||
|
$this->reorderTestsInSuite($_suite, $order, $resolveDependencies, $orderDefects);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($orderDefects === self::ORDER_DEFECTS_FIRST) {
|
||||||
|
$this->addSuiteToDefectSortOrder($suite);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->sort($suite, $order, $resolveDependencies, $orderDefects);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function sort(TestSuite $suite, int $order, bool $resolveDependencies, int $orderDefects): void
|
||||||
|
{
|
||||||
|
if ($suite->tests() === []) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($order === self::ORDER_REVERSED) {
|
||||||
|
$suite->setTests($this->reverse($suite->tests()));
|
||||||
|
} elseif ($order === self::ORDER_RANDOMIZED) {
|
||||||
|
$suite->setTests($this->randomize($suite->tests()));
|
||||||
|
} elseif ($order === self::ORDER_DURATION) {
|
||||||
|
$suite->setTests($this->sortByDuration($suite->tests()));
|
||||||
|
} elseif ($order === self::ORDER_SIZE) {
|
||||||
|
$suite->setTests($this->sortBySize($suite->tests()));
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($orderDefects === self::ORDER_DEFECTS_FIRST) {
|
||||||
|
$suite->setTests($this->sortDefectsFirst($suite->tests()));
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($resolveDependencies && ! ($suite instanceof DataProviderTestSuite)) {
|
||||||
|
$tests = $suite->tests();
|
||||||
|
|
||||||
|
/** @noinspection PhpParamsInspection */
|
||||||
|
/** @phpstan-ignore argument.type */
|
||||||
|
$suite->setTests($this->resolveDependencies($tests));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function addSuiteToDefectSortOrder(TestSuite $suite): void
|
||||||
|
{
|
||||||
|
$max = 0;
|
||||||
|
|
||||||
|
foreach ($suite->tests() as $test) {
|
||||||
|
assert($test instanceof Reorderable);
|
||||||
|
|
||||||
|
$sortId = $test->sortId();
|
||||||
|
|
||||||
|
if (! isset($this->defectSortOrder[$sortId])) {
|
||||||
|
$this->defectSortOrder[$sortId] = $this->cache->status(ResultCacheId::fromReorderable($test))->asInt();
|
||||||
|
$max = max($max, $this->defectSortOrder[$sortId]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->defectSortOrder[$suite->sortId()] = $max;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<Test> $tests
|
||||||
|
* @return list<Test>
|
||||||
|
*/
|
||||||
|
private function reverse(array $tests): array
|
||||||
|
{
|
||||||
|
return array_reverse($tests);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<Test> $tests
|
||||||
|
* @return list<Test>
|
||||||
|
*/
|
||||||
|
private function randomize(array $tests): array
|
||||||
|
{
|
||||||
|
shuffle($tests);
|
||||||
|
|
||||||
|
return $tests;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<Test> $tests
|
||||||
|
* @return list<Test>
|
||||||
|
*/
|
||||||
|
private function sortDefectsFirst(array $tests): array
|
||||||
|
{
|
||||||
|
usort(
|
||||||
|
$tests,
|
||||||
|
fn (Test $left, Test $right) => $this->cmpDefectPriorityAndTime($left, $right),
|
||||||
|
);
|
||||||
|
|
||||||
|
return $tests;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<Test> $tests
|
||||||
|
* @return list<Test>
|
||||||
|
*/
|
||||||
|
private function sortByDuration(array $tests): array
|
||||||
|
{
|
||||||
|
usort(
|
||||||
|
$tests,
|
||||||
|
fn (Test $left, Test $right) => $this->cmpDuration($left, $right),
|
||||||
|
);
|
||||||
|
|
||||||
|
return $tests;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<Test> $tests
|
||||||
|
* @return list<Test>
|
||||||
|
*/
|
||||||
|
private function sortBySize(array $tests): array
|
||||||
|
{
|
||||||
|
usort(
|
||||||
|
$tests,
|
||||||
|
fn (Test $left, Test $right) => $this->cmpSize($left, $right),
|
||||||
|
);
|
||||||
|
|
||||||
|
return $tests;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Comparator callback function to sort tests for "reach failure as fast as possible".
|
||||||
|
*
|
||||||
|
* 1. sort tests by defect weight defined in self::DEFECT_SORT_WEIGHT
|
||||||
|
* 2. when tests are equally defective, sort the fastest to the front
|
||||||
|
* 3. do not reorder successful tests
|
||||||
|
*/
|
||||||
|
private function cmpDefectPriorityAndTime(Test $a, Test $b): int
|
||||||
|
{
|
||||||
|
assert($a instanceof Reorderable);
|
||||||
|
assert($b instanceof Reorderable);
|
||||||
|
|
||||||
|
$priorityA = $this->defectSortOrder[$a->sortId()] ?? 0;
|
||||||
|
$priorityB = $this->defectSortOrder[$b->sortId()] ?? 0;
|
||||||
|
|
||||||
|
if ($priorityA !== $priorityB) {
|
||||||
|
// Sort defect weight descending
|
||||||
|
return $priorityB <=> $priorityA;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($priorityA > 0 || $priorityB > 0) {
|
||||||
|
return $this->cmpDuration($a, $b);
|
||||||
|
}
|
||||||
|
|
||||||
|
// do not change execution order
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compares test duration for sorting tests by duration ascending.
|
||||||
|
*/
|
||||||
|
private function cmpDuration(Test $a, Test $b): int
|
||||||
|
{
|
||||||
|
if (! ($a instanceof Reorderable && $b instanceof Reorderable)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->cache->time(ResultCacheId::fromReorderable($a)) <=> $this->cache->time(ResultCacheId::fromReorderable($b));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compares test size for sorting tests small->medium->large->unknown.
|
||||||
|
*/
|
||||||
|
private function cmpSize(Test $a, Test $b): int
|
||||||
|
{
|
||||||
|
$sizeA = ($a instanceof TestCase || $a instanceof DataProviderTestSuite)
|
||||||
|
? $a->size()->asString()
|
||||||
|
: 'unknown';
|
||||||
|
$sizeB = ($b instanceof TestCase || $b instanceof DataProviderTestSuite)
|
||||||
|
? $b->size()->asString()
|
||||||
|
: 'unknown';
|
||||||
|
|
||||||
|
return self::SIZE_SORT_WEIGHT[$sizeA] <=> self::SIZE_SORT_WEIGHT[$sizeB];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reorder Tests within a TestCase in such a way as to resolve as many dependencies as possible.
|
||||||
|
* The algorithm will leave the tests in original running order when it can.
|
||||||
|
* For more details see the documentation for test dependencies.
|
||||||
|
*
|
||||||
|
* Short description of algorithm:
|
||||||
|
* 1. Pick the next Test from remaining tests to be checked for dependencies.
|
||||||
|
* 2. If the test has no dependencies: mark done, start again from the top
|
||||||
|
* 3. If the test has dependencies but none left to do: mark done, start again from the top
|
||||||
|
* 4. When we reach the end add any leftover tests to the end. These will be marked 'skipped' during execution.
|
||||||
|
*
|
||||||
|
* @param array<TestCase> $tests
|
||||||
|
* @return array<TestCase>
|
||||||
|
*/
|
||||||
|
private function resolveDependencies(array $tests): array
|
||||||
|
{
|
||||||
|
// Pest: Fast-path. If no test in this suite declares dependencies, the
|
||||||
|
// original O(N^2) algorithm is wasted work — it would splice each test
|
||||||
|
// one-by-one back into the same order. The check deliberately walks
|
||||||
|
// TestCase instances directly instead of calling TestSuite::requires(),
|
||||||
|
// because the latter lazily builds TestSuite::provides() via
|
||||||
|
// ExecutionOrderDependency::mergeUnique, which is O(N^2) in the total
|
||||||
|
// number of tests. With thousands of tests that single call alone can
|
||||||
|
// burn several seconds before the sort even begins. Reading the
|
||||||
|
// cached TestCase::$dependencies property stays O(N) and costs nothing
|
||||||
|
// when no test uses `->depends()` / PHPUnit `@depends`.
|
||||||
|
if (! $this->anyTestHasDependencies($tests)) {
|
||||||
|
return $tests;
|
||||||
|
}
|
||||||
|
|
||||||
|
$newTestOrder = [];
|
||||||
|
$i = 0;
|
||||||
|
$provided = [];
|
||||||
|
|
||||||
|
do {
|
||||||
|
if (array_diff($tests[$i]->requires(), $provided) === []) {
|
||||||
|
$provided = array_merge($provided, $tests[$i]->provides());
|
||||||
|
$newTestOrder = array_merge($newTestOrder, array_splice($tests, $i, 1));
|
||||||
|
$i = 0;
|
||||||
|
} else {
|
||||||
|
$i++;
|
||||||
|
}
|
||||||
|
} while ($tests !== [] && ($i < count($tests)));
|
||||||
|
|
||||||
|
return array_merge($newTestOrder, $tests);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cheaply determines whether any test in the tree declares @depends.
|
||||||
|
*
|
||||||
|
* Walks `TestSuite` containers recursively and inspects each `TestCase`
|
||||||
|
* directly so it never triggers `TestSuite::provides()`, which is O(N^2)
|
||||||
|
* in the total number of aggregated tests.
|
||||||
|
*
|
||||||
|
* @param iterable<Test> $tests
|
||||||
|
*/
|
||||||
|
private function anyTestHasDependencies(iterable $tests): bool
|
||||||
|
{
|
||||||
|
foreach ($tests as $test) {
|
||||||
|
if ($test instanceof TestSuite) {
|
||||||
|
if ($this->anyTestHasDependencies($test->tests())) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($test instanceof TestCase && $test->requires() !== []) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -5,6 +5,8 @@
|
|||||||
[$bgBadgeColor, $bgBadgeText] = match ($type) {
|
[$bgBadgeColor, $bgBadgeText] = match ($type) {
|
||||||
'INFO' => ['blue', 'INFO'],
|
'INFO' => ['blue', 'INFO'],
|
||||||
'ERROR' => ['red', 'ERROR'],
|
'ERROR' => ['red', 'ERROR'],
|
||||||
|
'WARN' => ['yellow', 'WARN'],
|
||||||
|
'SUCCESS' => ['green', 'SUCCESS'],
|
||||||
};
|
};
|
||||||
|
|
||||||
?>
|
?>
|
||||||
|
|||||||
@ -176,9 +176,5 @@ final class Laravel extends AbstractPreset
|
|||||||
->toImplement('Illuminate\Contracts\Container\ContextualAttribute')
|
->toImplement('Illuminate\Contracts\Container\ContextualAttribute')
|
||||||
->toHaveAttribute('Attribute')
|
->toHaveAttribute('Attribute')
|
||||||
->toHaveMethod('resolve');
|
->toHaveMethod('resolve');
|
||||||
|
|
||||||
$this->expectations[] = expect('App\Rules')
|
|
||||||
->classes()
|
|
||||||
->toImplement('Illuminate\Contracts\Validation\ValidationRule');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -21,6 +21,7 @@ final class BootOverrides implements Bootstrapper
|
|||||||
'Runner/Filter/NameFilterIterator.php',
|
'Runner/Filter/NameFilterIterator.php',
|
||||||
'Runner/ResultCache/DefaultResultCache.php',
|
'Runner/ResultCache/DefaultResultCache.php',
|
||||||
'Runner/TestSuiteLoader.php',
|
'Runner/TestSuiteLoader.php',
|
||||||
|
'Runner/TestSuiteSorter.php',
|
||||||
'TextUI/Command/Commands/WarmCodeCoverageCacheCommand.php',
|
'TextUI/Command/Commands/WarmCodeCoverageCacheCommand.php',
|
||||||
'TextUI/Output/Default/ProgressPrinter/Subscriber/TestSkippedSubscriber.php',
|
'TextUI/Output/Default/ProgressPrinter/Subscriber/TestSkippedSubscriber.php',
|
||||||
'TextUI/TestSuiteFilterProcessor.php',
|
'TextUI/TestSuiteFilterProcessor.php',
|
||||||
|
|||||||
19
src/Bootstrappers/BootPhpUnitConfiguration.php
Normal file
19
src/Bootstrappers/BootPhpUnitConfiguration.php
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Bootstrappers;
|
||||||
|
|
||||||
|
use Pest\Contracts\Bootstrapper;
|
||||||
|
use PHPUnit\TextUI\Configuration\Builder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class BootPhpUnitConfiguration implements Bootstrapper
|
||||||
|
{
|
||||||
|
public function boot(): void
|
||||||
|
{
|
||||||
|
(new Builder)->build(['pest']);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -25,6 +25,17 @@ final readonly class BootSubscribers implements Bootstrapper
|
|||||||
Subscribers\EnsureIgnorableTestCasesAreIgnored::class,
|
Subscribers\EnsureIgnorableTestCasesAreIgnored::class,
|
||||||
Subscribers\EnsureKernelDumpIsFlushed::class,
|
Subscribers\EnsureKernelDumpIsFlushed::class,
|
||||||
Subscribers\EnsureTeamCityEnabled::class,
|
Subscribers\EnsureTeamCityEnabled::class,
|
||||||
|
Subscribers\EnsureTiaIsRunningPestTestsOnly::class,
|
||||||
|
Subscribers\EnsureTiaStarts::class,
|
||||||
|
Subscribers\EnsureTiaEnds::class,
|
||||||
|
Subscribers\EnsureTiaResultsAreCollected::class,
|
||||||
|
Subscribers\EnsureTiaResultIsRecordedOnPassed::class,
|
||||||
|
Subscribers\EnsureTiaResultIsRecordedOnFailed::class,
|
||||||
|
Subscribers\EnsureTiaResultIsRecordedOnErrored::class,
|
||||||
|
Subscribers\EnsureTiaResultIsRecordedOnSkipped::class,
|
||||||
|
Subscribers\EnsureTiaResultIsRecordedOnIncomplete::class,
|
||||||
|
Subscribers\EnsureTiaResultIsRecordedOnRisky::class,
|
||||||
|
Subscribers\EnsureTiaAssertionsAreRecordedOnFinished::class,
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@ -7,12 +7,18 @@ namespace Pest\Concerns;
|
|||||||
use Closure;
|
use Closure;
|
||||||
use Pest\Exceptions\DatasetArgumentsMismatch;
|
use Pest\Exceptions\DatasetArgumentsMismatch;
|
||||||
use Pest\Panic;
|
use Pest\Panic;
|
||||||
|
use Pest\Plugins\Tia;
|
||||||
|
use Pest\Plugins\Tia\Collectors;
|
||||||
|
use Pest\Plugins\Tia\Enums\ReplayType;
|
||||||
|
use Pest\Plugins\Tia\Recorder;
|
||||||
use Pest\Preset;
|
use Pest\Preset;
|
||||||
use Pest\Support\ChainableClosure;
|
use Pest\Support\ChainableClosure;
|
||||||
|
use Pest\Support\Container;
|
||||||
use Pest\Support\ExceptionTrace;
|
use Pest\Support\ExceptionTrace;
|
||||||
use Pest\Support\Reflection;
|
use Pest\Support\Reflection;
|
||||||
use Pest\Support\Shell;
|
use Pest\Support\Shell;
|
||||||
use Pest\TestSuite;
|
use Pest\TestSuite;
|
||||||
|
use PHPUnit\Framework\AssertionFailedError;
|
||||||
use PHPUnit\Framework\Attributes\PostCondition;
|
use PHPUnit\Framework\Attributes\PostCondition;
|
||||||
use PHPUnit\Framework\IncompleteTest;
|
use PHPUnit\Framework\IncompleteTest;
|
||||||
use PHPUnit\Framework\SkippedTest;
|
use PHPUnit\Framework\SkippedTest;
|
||||||
@ -75,6 +81,17 @@ trait Testable
|
|||||||
*/
|
*/
|
||||||
public bool $__ran = false;
|
public bool $__ran = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The active replay mode for this test, set in `setUp()` and checked
|
||||||
|
* in `__runTest()` / `tearDown()` to skip the body and after-each.
|
||||||
|
*/
|
||||||
|
private ReplayType $__replay = ReplayType::None;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The cached assertion count to replay, captured when entering replay mode.
|
||||||
|
*/
|
||||||
|
private int $__replayAssertions = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The test's test closure.
|
* The test's test closure.
|
||||||
*/
|
*/
|
||||||
@ -259,8 +276,35 @@ trait Testable
|
|||||||
self::$__latestIssues = $method->issues;
|
self::$__latestIssues = $method->issues;
|
||||||
self::$__latestPrs = $method->prs;
|
self::$__latestPrs = $method->prs;
|
||||||
|
|
||||||
|
/** @var Tia $tia */
|
||||||
|
$tia = Container::getInstance()->get(Tia::class);
|
||||||
|
$status = $tia->getStatus(self::$__filename, $this::class.'::'.$this->name());
|
||||||
|
$replay = ReplayType::fromStatus($status);
|
||||||
|
|
||||||
|
if ($replay !== ReplayType::None) {
|
||||||
|
assert($status !== null);
|
||||||
|
|
||||||
|
match ($replay) {
|
||||||
|
ReplayType::Pass, ReplayType::Risky => $this->__beginReplay($replay, $tia),
|
||||||
|
ReplayType::Skipped => $this->markTestSkipped($status->message()),
|
||||||
|
ReplayType::Incomplete => $this->markTestIncomplete($status->message()),
|
||||||
|
ReplayType::Failure => throw new AssertionFailedError($status->message() ?: 'Cached failure'),
|
||||||
|
};
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$recorder = Container::getInstance()->get(Recorder::class);
|
||||||
|
assert($recorder instanceof Recorder);
|
||||||
|
|
||||||
|
if ($recorder->isActive()) {
|
||||||
|
$recorder->beginTest($this::class, $this->name(), self::$__filename);
|
||||||
|
}
|
||||||
|
|
||||||
parent::setUp();
|
parent::setUp();
|
||||||
|
|
||||||
|
Collectors::armAll($recorder);
|
||||||
|
|
||||||
$beforeEach = TestSuite::getInstance()->beforeEach->get(self::$__filename)[1];
|
$beforeEach = TestSuite::getInstance()->beforeEach->get(self::$__filename)[1];
|
||||||
|
|
||||||
if ($this->__beforeEach instanceof Closure) {
|
if ($this->__beforeEach instanceof Closure) {
|
||||||
@ -270,6 +314,13 @@ trait Testable
|
|||||||
$this->__callClosure($beforeEach, $arguments);
|
$this->__callClosure($beforeEach, $arguments);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function __beginReplay(ReplayType $replay, Tia $tia): void
|
||||||
|
{
|
||||||
|
$this->__replay = $replay;
|
||||||
|
$this->__replayAssertions = $tia->getAssertionCount($this::class.'::'.$this->name());
|
||||||
|
$this->__ran = true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize test case properties from TestSuite.
|
* Initialize test case properties from TestSuite.
|
||||||
*/
|
*/
|
||||||
@ -302,6 +353,12 @@ trait Testable
|
|||||||
*/
|
*/
|
||||||
protected function tearDown(...$arguments): void
|
protected function tearDown(...$arguments): void
|
||||||
{
|
{
|
||||||
|
if ($this->__replay !== ReplayType::None) {
|
||||||
|
TestSuite::getInstance()->test = null;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
$afterEach = TestSuite::getInstance()->afterEach->get(self::$__filename);
|
$afterEach = TestSuite::getInstance()->afterEach->get(self::$__filename);
|
||||||
|
|
||||||
if ($this->__afterEach instanceof Closure) {
|
if ($this->__afterEach instanceof Closure) {
|
||||||
@ -327,6 +384,16 @@ trait Testable
|
|||||||
*/
|
*/
|
||||||
private function __runTest(Closure $closure, ...$args): mixed
|
private function __runTest(Closure $closure, ...$args): mixed
|
||||||
{
|
{
|
||||||
|
if ($this->__replay === ReplayType::Pass || $this->__replay === ReplayType::Risky) {
|
||||||
|
if ($this->__replay === ReplayType::Pass && $this->__replayAssertions === 0) {
|
||||||
|
$this->expectNotToPerformAssertions();
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->addToAssertionCount($this->__replayAssertions);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
$arguments = $this->__resolveTestArguments($args);
|
$arguments = $this->__resolveTestArguments($args);
|
||||||
$this->__ensureDatasetArgumentNameAndNumberMatches($arguments);
|
$this->__ensureDatasetArgumentNameAndNumberMatches($arguments);
|
||||||
|
|
||||||
|
|||||||
@ -119,6 +119,14 @@ final readonly class Configuration
|
|||||||
return new Browser\Configuration;
|
return new Browser\Configuration;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the TIA (Test Impact Analysis) configuration.
|
||||||
|
*/
|
||||||
|
public function tia(): Plugins\Tia\Configuration
|
||||||
|
{
|
||||||
|
return new Plugins\Tia\Configuration;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Proxies calls to the uses method.
|
* Proxies calls to the uses method.
|
||||||
*
|
*
|
||||||
|
|||||||
16
src/Contracts/Restarter.php
Normal file
16
src/Contracts/Restarter.php
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Contracts;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
interface Restarter
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $arguments
|
||||||
|
*/
|
||||||
|
public function maybeRestart(string $projectRoot, array $arguments): void;
|
||||||
|
}
|
||||||
54
src/Exceptions/BaselineFetchFailed.php
Normal file
54
src/Exceptions/BaselineFetchFailed.php
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Exceptions;
|
||||||
|
|
||||||
|
use NunoMaduro\Collision\Contracts\RenderlessEditor;
|
||||||
|
use NunoMaduro\Collision\Contracts\RenderlessTrace;
|
||||||
|
use Pest\Contracts\Panicable;
|
||||||
|
use Pest\Support\View;
|
||||||
|
use RuntimeException;
|
||||||
|
use Symfony\Component\Console\Exception\ExceptionInterface;
|
||||||
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class BaselineFetchFailed extends RuntimeException implements ExceptionInterface, Panicable, RenderlessEditor, RenderlessTrace
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private readonly string $headline,
|
||||||
|
private readonly string $hint,
|
||||||
|
private readonly bool $hasAnchor = false,
|
||||||
|
) {
|
||||||
|
parent::__construct($headline);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function render(OutputInterface $output): void
|
||||||
|
{
|
||||||
|
View::renderUsing($output);
|
||||||
|
|
||||||
|
if (! $this->hasAnchor) {
|
||||||
|
View::render('components.badge', ['type' => 'ERROR', 'content' => $this->headline]);
|
||||||
|
$this->renderChild($output, $this->hint.' Or use [--fresh] to record locally.');
|
||||||
|
$output->writeln('');
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->renderChild($output, $this->headline);
|
||||||
|
$this->renderChild($output, $this->hint.' Or use [--fresh] to record locally.');
|
||||||
|
$output->writeln('');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function exitCode(): int
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function renderChild(OutputInterface $output, string $text): void
|
||||||
|
{
|
||||||
|
$output->writeln(sprintf(' <fg=gray>─ %s</>', $text));
|
||||||
|
}
|
||||||
|
}
|
||||||
32
src/Exceptions/NoAffectedTestsFound.php
Normal file
32
src/Exceptions/NoAffectedTestsFound.php
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Exceptions;
|
||||||
|
|
||||||
|
use InvalidArgumentException;
|
||||||
|
use NunoMaduro\Collision\Contracts\RenderlessEditor;
|
||||||
|
use NunoMaduro\Collision\Contracts\RenderlessTrace;
|
||||||
|
use Pest\Contracts\Panicable;
|
||||||
|
use Symfony\Component\Console\Exception\ExceptionInterface;
|
||||||
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class NoAffectedTestsFound extends InvalidArgumentException implements ExceptionInterface, Panicable, RenderlessEditor, RenderlessTrace
|
||||||
|
{
|
||||||
|
public function render(OutputInterface $output): void
|
||||||
|
{
|
||||||
|
$output->writeln([
|
||||||
|
'',
|
||||||
|
' <fg=white;options=bold;bg=blue> INFO </> No affected tests found.',
|
||||||
|
'',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function exitCode(): int
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
46
src/Exceptions/TiaRequiresPestTests.php
Normal file
46
src/Exceptions/TiaRequiresPestTests.php
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Exceptions;
|
||||||
|
|
||||||
|
use NunoMaduro\Collision\Contracts\RenderlessEditor;
|
||||||
|
use NunoMaduro\Collision\Contracts\RenderlessTrace;
|
||||||
|
use Pest\Contracts\Panicable;
|
||||||
|
use RuntimeException;
|
||||||
|
use Symfony\Component\Console\Exception\ExceptionInterface;
|
||||||
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class TiaRequiresPestTests extends RuntimeException implements ExceptionInterface, Panicable, RenderlessEditor, RenderlessTrace
|
||||||
|
{
|
||||||
|
public function __construct(private readonly string $className, string $filename)
|
||||||
|
{
|
||||||
|
parent::__construct(sprintf(
|
||||||
|
'Tia mode requires only functional based Pest tests, but encountered PHPUnit class [%s] in [%s].',
|
||||||
|
$className,
|
||||||
|
$filename,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function render(OutputInterface $output): void
|
||||||
|
{
|
||||||
|
$output->writeln([
|
||||||
|
'',
|
||||||
|
' <fg=white;options=bold;bg=red> ERROR </> Tia mode requires Pest tests.',
|
||||||
|
'',
|
||||||
|
sprintf(' Encountered PHPUnit class <fg=yellow>%s</>', $this->className),
|
||||||
|
sprintf(' in <fg=gray>%s</>.', $this->file),
|
||||||
|
'',
|
||||||
|
' Convert it to a Pest test, or run without Tia.',
|
||||||
|
'',
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function exitCode(): int
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -166,7 +166,7 @@ final class TestCaseFactory
|
|||||||
final class $className extends $baseClass implements $hasPrintableTestCaseClassFQN {
|
final class $className extends $baseClass implements $hasPrintableTestCaseClassFQN {
|
||||||
$traitsCode
|
$traitsCode
|
||||||
|
|
||||||
private static \$__filename = '$filename';
|
public static \$__filename = '$filename';
|
||||||
|
|
||||||
$methodsCode
|
$methodsCode
|
||||||
}
|
}
|
||||||
|
|||||||
@ -27,8 +27,13 @@ use Whoops\Exception\Inspector;
|
|||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
final readonly class Kernel
|
final class Kernel
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* Either the kernel is terminated or not.
|
||||||
|
*/
|
||||||
|
private bool $terminated = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The Kernel bootstrappers.
|
* The Kernel bootstrappers.
|
||||||
*
|
*
|
||||||
@ -36,6 +41,8 @@ final readonly class Kernel
|
|||||||
*/
|
*/
|
||||||
private const array BOOTSTRAPPERS = [
|
private const array BOOTSTRAPPERS = [
|
||||||
Bootstrappers\BootOverrides::class,
|
Bootstrappers\BootOverrides::class,
|
||||||
|
Bootstrappers\BootPhpUnitConfiguration::class,
|
||||||
|
Plugins\Tia\Bootstrapper::class,
|
||||||
Bootstrappers\BootSubscribers::class,
|
Bootstrappers\BootSubscribers::class,
|
||||||
Bootstrappers\BootFiles::class,
|
Bootstrappers\BootFiles::class,
|
||||||
Bootstrappers\BootView::class,
|
Bootstrappers\BootView::class,
|
||||||
@ -43,15 +50,22 @@ final readonly class Kernel
|
|||||||
Bootstrappers\BootExcludeList::class,
|
Bootstrappers\BootExcludeList::class,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The Kernel restarters — resolved and invoked from `bin/pest`
|
||||||
|
* before any other Pest class is touched, so the list is exposed
|
||||||
|
* on the Kernel rather than driven from `bin/pest` directly.
|
||||||
|
*
|
||||||
|
* @var array<int, class-string<Contracts\Restarter>>
|
||||||
|
*/
|
||||||
|
public const array RESTARTERS = [
|
||||||
|
Restarters\XdebugRestarter::class,
|
||||||
|
Restarters\PcovRestarter::class,
|
||||||
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new Kernel instance.
|
* Creates a new Kernel instance.
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(private readonly Application $application, private readonly OutputInterface $output) {}
|
||||||
private Application $application,
|
|
||||||
private OutputInterface $output,
|
|
||||||
) {
|
|
||||||
//
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Boots the Kernel.
|
* Boots the Kernel.
|
||||||
@ -112,9 +126,13 @@ final readonly class Kernel
|
|||||||
$configuration = Registry::get();
|
$configuration = Registry::get();
|
||||||
$result = Facade::result();
|
$result = Facade::result();
|
||||||
|
|
||||||
return CallsAddsOutput::execute(
|
$result = CallsAddsOutput::execute(
|
||||||
Result::exitCode($configuration, $result),
|
Result::exitCode($configuration, $result),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
$this->terminate();
|
||||||
|
|
||||||
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -122,6 +140,12 @@ final readonly class Kernel
|
|||||||
*/
|
*/
|
||||||
public function terminate(): void
|
public function terminate(): void
|
||||||
{
|
{
|
||||||
|
if ($this->terminated) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->terminated = true;
|
||||||
|
|
||||||
$preBufferOutput = Container::getInstance()->get(KernelDump::class);
|
$preBufferOutput = Container::getInstance()->get(KernelDump::class);
|
||||||
|
|
||||||
assert($preBufferOutput instanceof KernelDump);
|
assert($preBufferOutput instanceof KernelDump);
|
||||||
@ -139,7 +163,7 @@ final readonly class Kernel
|
|||||||
$this->terminate();
|
$this->terminate();
|
||||||
|
|
||||||
if (is_array($error = error_get_last())) {
|
if (is_array($error = error_get_last())) {
|
||||||
if (! in_array($error['type'], [E_ERROR, E_CORE_ERROR], true)) {
|
if (! in_array($error['type'], [E_ERROR, E_COMPILE_ERROR, E_CORE_ERROR], true)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -68,6 +68,10 @@ final class KernelDump
|
|||||||
|
|
||||||
$type = 'INFO';
|
$type = 'INFO';
|
||||||
|
|
||||||
|
if (is_array($error = error_get_last()) && in_array($error['type'], [E_ERROR, E_COMPILE_ERROR, E_CORE_ERROR], true)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if ($this->isInternalError($this->buffer)) {
|
if ($this->isInternalError($this->buffer)) {
|
||||||
$type = 'ERROR';
|
$type = 'ERROR';
|
||||||
$this->buffer = str_replace(
|
$this->buffer = str_replace(
|
||||||
@ -107,7 +111,6 @@ final class KernelDump
|
|||||||
*/
|
*/
|
||||||
private function isInternalError(string $output): bool
|
private function isInternalError(string $output): bool
|
||||||
{
|
{
|
||||||
return str_contains($output, 'An error occurred inside PHPUnit.')
|
return str_contains($output, 'An error occurred inside PHPUnit.');
|
||||||
|| str_contains($output, 'Fatal error');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -12,7 +12,9 @@ use PHPUnit\Event\Code\Test;
|
|||||||
use PHPUnit\Event\Code\TestMethod;
|
use PHPUnit\Event\Code\TestMethod;
|
||||||
use PHPUnit\Event\Code\Throwable;
|
use PHPUnit\Event\Code\Throwable;
|
||||||
use PHPUnit\Event\Test\AfterLastTestMethodErrored;
|
use PHPUnit\Event\Test\AfterLastTestMethodErrored;
|
||||||
|
use PHPUnit\Event\Test\AfterLastTestMethodFailed;
|
||||||
use PHPUnit\Event\Test\BeforeFirstTestMethodErrored;
|
use PHPUnit\Event\Test\BeforeFirstTestMethodErrored;
|
||||||
|
use PHPUnit\Event\Test\BeforeFirstTestMethodFailed;
|
||||||
use PHPUnit\Event\Test\ConsideredRisky;
|
use PHPUnit\Event\Test\ConsideredRisky;
|
||||||
use PHPUnit\Event\Test\Errored;
|
use PHPUnit\Event\Test\Errored;
|
||||||
use PHPUnit\Event\Test\Failed;
|
use PHPUnit\Event\Test\Failed;
|
||||||
@ -255,9 +257,11 @@ final readonly class Converter
|
|||||||
$numberOfNotPassedTests = count(
|
$numberOfNotPassedTests = count(
|
||||||
array_unique(
|
array_unique(
|
||||||
array_map(
|
array_map(
|
||||||
function (AfterLastTestMethodErrored|BeforeFirstTestMethodErrored|Errored|Failed|Skipped|ConsideredRisky|MarkedIncomplete $event): string {
|
function (AfterLastTestMethodErrored|AfterLastTestMethodFailed|BeforeFirstTestMethodErrored|BeforeFirstTestMethodFailed|Errored|Failed|Skipped|ConsideredRisky|MarkedIncomplete $event): string {
|
||||||
if ($event instanceof BeforeFirstTestMethodErrored
|
if ($event instanceof BeforeFirstTestMethodErrored
|
||||||
|| $event instanceof AfterLastTestMethodErrored) {
|
|| $event instanceof AfterLastTestMethodErrored
|
||||||
|
|| $event instanceof BeforeFirstTestMethodFailed
|
||||||
|
|| $event instanceof AfterLastTestMethodFailed) {
|
||||||
return $event->testClassName();
|
return $event->testClassName();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -14,6 +14,7 @@ use InvalidArgumentException;
|
|||||||
use JsonSerializable;
|
use JsonSerializable;
|
||||||
use Pest\Exceptions\InvalidExpectationValue;
|
use Pest\Exceptions\InvalidExpectationValue;
|
||||||
use Pest\Matchers\Any;
|
use Pest\Matchers\Any;
|
||||||
|
use Pest\Plugins\Snapshot;
|
||||||
use Pest\Support\Arr;
|
use Pest\Support\Arr;
|
||||||
use Pest\Support\Exporter;
|
use Pest\Support\Exporter;
|
||||||
use Pest\Support\NullClosure;
|
use Pest\Support\NullClosure;
|
||||||
@ -851,18 +852,31 @@ final class Expectation
|
|||||||
default => InvalidExpectationValue::expected('array|object|string'),
|
default => InvalidExpectationValue::expected('array|object|string'),
|
||||||
};
|
};
|
||||||
|
|
||||||
if ($snapshots->has()) {
|
if (! $snapshots->has()) {
|
||||||
[$filename, $content] = $snapshots->get();
|
|
||||||
|
|
||||||
Assert::assertSame(
|
|
||||||
strtr($content, ["\r\n" => "\n", "\r" => "\n"]),
|
|
||||||
strtr($string, ["\r\n" => "\n", "\r" => "\n"]),
|
|
||||||
$message === '' ? "Failed asserting that the string value matches its snapshot ($filename)." : $message
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
$filename = $snapshots->save($string);
|
$filename = $snapshots->save($string);
|
||||||
|
|
||||||
TestSuite::getInstance()->registerSnapshotChange("Snapshot created at [$filename]");
|
TestSuite::getInstance()->registerSnapshotChange("Snapshot created at [$filename]");
|
||||||
|
} else {
|
||||||
|
[$filename, $content] = $snapshots->get();
|
||||||
|
|
||||||
|
$normalizedContent = strtr($content, ["\r\n" => "\n", "\r" => "\n"]);
|
||||||
|
$normalizedString = strtr($string, ["\r\n" => "\n", "\r" => "\n"]);
|
||||||
|
|
||||||
|
if (Snapshot::$updateSnapshots && $normalizedContent !== $normalizedString) {
|
||||||
|
$snapshots->save($string);
|
||||||
|
|
||||||
|
TestSuite::getInstance()->registerSnapshotChange("Snapshot updated at [$filename]");
|
||||||
|
} else {
|
||||||
|
if (Snapshot::$updateSnapshots) {
|
||||||
|
TestSuite::getInstance()->registerSnapshotChange("Snapshot unchanged at [$filename]");
|
||||||
|
}
|
||||||
|
|
||||||
|
Assert::assertSame(
|
||||||
|
$normalizedContent,
|
||||||
|
$normalizedString,
|
||||||
|
$message === '' ? "Failed asserting that the string value matches its snapshot ($filename)." : $message
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
@ -940,6 +954,7 @@ final class Expectation
|
|||||||
} catch (Throwable $e) {
|
} catch (Throwable $e) {
|
||||||
|
|
||||||
if ($exception instanceof Throwable) {
|
if ($exception instanceof Throwable) {
|
||||||
|
// @phpstan-ignore-next-line
|
||||||
expect($e)
|
expect($e)
|
||||||
->toBeInstanceOf($exception::class, $message)
|
->toBeInstanceOf($exception::class, $message)
|
||||||
->and($e->getMessage())->toBe($exceptionMessage ?? $exception->getMessage(), $message);
|
->and($e->getMessage())->toBe($exceptionMessage ?? $exception->getMessage(), $message);
|
||||||
|
|||||||
@ -6,7 +6,7 @@ namespace Pest;
|
|||||||
|
|
||||||
function version(): string
|
function version(): string
|
||||||
{
|
{
|
||||||
return '4.4.4';
|
return '4.7.1';
|
||||||
}
|
}
|
||||||
|
|
||||||
function testDirectory(string $file = ''): string
|
function testDirectory(string $file = ''): string
|
||||||
|
|||||||
@ -123,6 +123,10 @@ final readonly class Help implements HandlesArguments
|
|||||||
'arg' => '--update-snapshots',
|
'arg' => '--update-snapshots',
|
||||||
'desc' => 'Update snapshots for tests using the "toMatchSnapshot" expectation',
|
'desc' => 'Update snapshots for tests using the "toMatchSnapshot" expectation',
|
||||||
],
|
],
|
||||||
|
[
|
||||||
|
'arg' => '--update-shards',
|
||||||
|
'desc' => 'Update shards.json with test timing data for time-balanced sharding',
|
||||||
|
],
|
||||||
], ...$content['Execution']];
|
], ...$content['Execution']];
|
||||||
|
|
||||||
$content['Selection'] = [[
|
$content['Selection'] = [[
|
||||||
|
|||||||
@ -146,7 +146,6 @@ final class WrapperRunner implements RunnerInterface
|
|||||||
public function run(): int
|
public function run(): int
|
||||||
{
|
{
|
||||||
$directory = dirname(__DIR__);
|
$directory = dirname(__DIR__);
|
||||||
assert($directory !== '');
|
|
||||||
ExcludeList::addDirectory($directory);
|
ExcludeList::addDirectory($directory);
|
||||||
TestResultFacade::init();
|
TestResultFacade::init();
|
||||||
EventFacade::instance()->seal();
|
EventFacade::instance()->seal();
|
||||||
|
|||||||
@ -6,7 +6,13 @@ namespace Pest\Plugins;
|
|||||||
|
|
||||||
use Pest\Contracts\Plugins\AddsOutput;
|
use Pest\Contracts\Plugins\AddsOutput;
|
||||||
use Pest\Contracts\Plugins\HandlesArguments;
|
use Pest\Contracts\Plugins\HandlesArguments;
|
||||||
|
use Pest\Contracts\Plugins\Terminable;
|
||||||
use Pest\Exceptions\InvalidOption;
|
use Pest\Exceptions\InvalidOption;
|
||||||
|
use Pest\Subscribers\EnsureShardTimingFinished;
|
||||||
|
use Pest\Subscribers\EnsureShardTimingsAreCollected;
|
||||||
|
use Pest\Subscribers\EnsureShardTimingStarted;
|
||||||
|
use Pest\TestSuite;
|
||||||
|
use PHPUnit\Event;
|
||||||
use Symfony\Component\Console\Input\ArgvInput;
|
use Symfony\Component\Console\Input\ArgvInput;
|
||||||
use Symfony\Component\Console\Input\InputInterface;
|
use Symfony\Component\Console\Input\InputInterface;
|
||||||
use Symfony\Component\Console\Output\OutputInterface;
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
@ -15,7 +21,7 @@ use Symfony\Component\Process\Process;
|
|||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
final class Shard implements AddsOutput, HandlesArguments
|
final class Shard implements AddsOutput, HandlesArguments, Terminable
|
||||||
{
|
{
|
||||||
use Concerns\HandleArguments;
|
use Concerns\HandleArguments;
|
||||||
|
|
||||||
@ -33,6 +39,40 @@ final class Shard implements AddsOutput, HandlesArguments
|
|||||||
*/
|
*/
|
||||||
private static ?array $shard = null;
|
private static ?array $shard = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether to update the shards.json file.
|
||||||
|
*/
|
||||||
|
private static bool $updateShards = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether time-balanced sharding was used.
|
||||||
|
*/
|
||||||
|
private static bool $timeBalanced = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether the shards.json file is outdated.
|
||||||
|
*/
|
||||||
|
private static bool $shardsOutdated = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether the test suite passed.
|
||||||
|
*/
|
||||||
|
private static bool $passed = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Collected timings from workers or subscribers.
|
||||||
|
*
|
||||||
|
* @var array<string, float>|null
|
||||||
|
*/
|
||||||
|
private static ?array $collectedTimings = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The canonical list of test classes from --list-tests.
|
||||||
|
*
|
||||||
|
* @var list<string>|null
|
||||||
|
*/
|
||||||
|
private static ?array $knownTests = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new Plugin instance.
|
* Creates a new Plugin instance.
|
||||||
*/
|
*/
|
||||||
@ -47,6 +87,19 @@ final class Shard implements AddsOutput, HandlesArguments
|
|||||||
*/
|
*/
|
||||||
public function handleArguments(array $arguments): array
|
public function handleArguments(array $arguments): array
|
||||||
{
|
{
|
||||||
|
if ($this->hasArgument('--update-shards', $arguments)) {
|
||||||
|
return $this->handleUpdateShards($arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Parallel::isWorker() && Parallel::getGlobal('UPDATE_SHARDS') === true) {
|
||||||
|
self::$updateShards = true;
|
||||||
|
|
||||||
|
Event\Facade::instance()->registerSubscriber(new EnsureShardTimingStarted);
|
||||||
|
Event\Facade::instance()->registerSubscriber(new EnsureShardTimingFinished);
|
||||||
|
|
||||||
|
return $arguments;
|
||||||
|
}
|
||||||
|
|
||||||
if (! $this->hasArgument('--shard', $arguments)) {
|
if (! $this->hasArgument('--shard', $arguments)) {
|
||||||
return $arguments;
|
return $arguments;
|
||||||
}
|
}
|
||||||
@ -63,7 +116,24 @@ final class Shard implements AddsOutput, HandlesArguments
|
|||||||
|
|
||||||
/** @phpstan-ignore-next-line */
|
/** @phpstan-ignore-next-line */
|
||||||
$tests = $this->allTests($arguments);
|
$tests = $this->allTests($arguments);
|
||||||
|
|
||||||
|
$timings = $this->loadShardsFile();
|
||||||
|
if ($timings !== null) {
|
||||||
|
$knownTests = array_values(array_filter($tests, fn (string $test): bool => isset($timings[$test])));
|
||||||
|
$newTests = array_values(array_diff($tests, $knownTests));
|
||||||
|
|
||||||
|
$partitions = $this->partitionByTime($knownTests, $timings, $total);
|
||||||
|
|
||||||
|
foreach ($newTests as $i => $test) {
|
||||||
|
$partitions[$i % $total][] = $test;
|
||||||
|
}
|
||||||
|
|
||||||
|
$testsToRun = $partitions[$index - 1] ?? [];
|
||||||
|
self::$timeBalanced = true;
|
||||||
|
self::$shardsOutdated = $newTests !== [];
|
||||||
|
} else {
|
||||||
$testsToRun = (array_chunk($tests, max(1, (int) ceil(count($tests) / $total))))[$index - 1] ?? [];
|
$testsToRun = (array_chunk($tests, max(1, (int) ceil(count($tests) / $total))))[$index - 1] ?? [];
|
||||||
|
}
|
||||||
|
|
||||||
self::$shard = [
|
self::$shard = [
|
||||||
'index' => $index,
|
'index' => $index,
|
||||||
@ -72,9 +142,43 @@ final class Shard implements AddsOutput, HandlesArguments
|
|||||||
'testsCount' => count($tests),
|
'testsCount' => count($tests),
|
||||||
];
|
];
|
||||||
|
|
||||||
|
if ($testsToRun === []) {
|
||||||
|
return $arguments;
|
||||||
|
}
|
||||||
|
|
||||||
return [...$arguments, '--filter', $this->buildFilterArgument($testsToRun)];
|
return [...$arguments, '--filter', $this->buildFilterArgument($testsToRun)];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handles the --update-shards argument.
|
||||||
|
*
|
||||||
|
* @param array<int, string> $arguments
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private function handleUpdateShards(array $arguments): array
|
||||||
|
{
|
||||||
|
if ($this->hasArgument('--shard', $arguments)) {
|
||||||
|
throw new InvalidOption('The [--update-shards] option cannot be combined with [--shard].');
|
||||||
|
}
|
||||||
|
|
||||||
|
$arguments = $this->popArgument('--update-shards', $arguments);
|
||||||
|
|
||||||
|
self::$updateShards = true;
|
||||||
|
|
||||||
|
/** @phpstan-ignore-next-line */
|
||||||
|
self::$knownTests = $this->allTests($arguments);
|
||||||
|
|
||||||
|
if ($this->hasArgument('--parallel', $arguments) || $this->hasArgument('-p', $arguments)) {
|
||||||
|
Parallel::setGlobal('UPDATE_SHARDS', true);
|
||||||
|
Parallel::setGlobal('SHARD_RUN_ID', uniqid('pest-shard-', true));
|
||||||
|
} else {
|
||||||
|
Event\Facade::instance()->registerSubscriber(new EnsureShardTimingStarted);
|
||||||
|
Event\Facade::instance()->registerSubscriber(new EnsureShardTimingFinished);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $arguments;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns all tests that the test suite would run.
|
* Returns all tests that the test suite would run.
|
||||||
*
|
*
|
||||||
@ -87,7 +191,7 @@ final class Shard implements AddsOutput, HandlesArguments
|
|||||||
'php',
|
'php',
|
||||||
...$this->removeParallelArguments($arguments),
|
...$this->removeParallelArguments($arguments),
|
||||||
'--list-tests',
|
'--list-tests',
|
||||||
]))->mustRun()->getOutput();
|
]))->setTimeout(120)->mustRun()->getOutput();
|
||||||
|
|
||||||
preg_match_all('/ - (?:P\\\\)?(Tests\\\\[^:]+)::/', $output, $matches);
|
preg_match_all('/ - (?:P\\\\)?(Tests\\\\[^:]+)::/', $output, $matches);
|
||||||
|
|
||||||
@ -116,6 +220,22 @@ final class Shard implements AddsOutput, HandlesArguments
|
|||||||
*/
|
*/
|
||||||
public function addOutput(int $exitCode): int
|
public function addOutput(int $exitCode): int
|
||||||
{
|
{
|
||||||
|
self::$passed = $exitCode === 0;
|
||||||
|
|
||||||
|
if (self::$updateShards && self::$passed && ! Parallel::isWorker()) {
|
||||||
|
self::$collectedTimings = $this->collectTimings();
|
||||||
|
|
||||||
|
$count = self::$knownTests !== null
|
||||||
|
? count(array_intersect_key(self::$collectedTimings, array_flip(self::$knownTests)))
|
||||||
|
: count(self::$collectedTimings);
|
||||||
|
|
||||||
|
$this->output->writeln(sprintf(
|
||||||
|
' <fg=gray>Shards:</> <fg=default>shards.json updated with timings for %d test class%s.</>',
|
||||||
|
$count,
|
||||||
|
$count === 1 ? '' : 'es',
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
if (self::$shard === null) {
|
if (self::$shard === null) {
|
||||||
return $exitCode;
|
return $exitCode;
|
||||||
}
|
}
|
||||||
@ -128,17 +248,250 @@ final class Shard implements AddsOutput, HandlesArguments
|
|||||||
] = self::$shard;
|
] = self::$shard;
|
||||||
|
|
||||||
$this->output->writeln(sprintf(
|
$this->output->writeln(sprintf(
|
||||||
' <fg=gray>Shard:</> <fg=default>%d of %d</> — %d file%s ran, out of %d.',
|
' <fg=gray>Shard:</> <fg=default>%d of %d</> — %d file%s ran, out of %d%s.',
|
||||||
$index,
|
$index,
|
||||||
$total,
|
$total,
|
||||||
$testsRan,
|
$testsRan,
|
||||||
$testsRan === 1 ? '' : 's',
|
$testsRan === 1 ? '' : 's',
|
||||||
$testsCount,
|
$testsCount,
|
||||||
|
self::$timeBalanced ? ' <fg=gray>(time-balanced)</>' : '',
|
||||||
));
|
));
|
||||||
|
|
||||||
|
if (self::$shardsOutdated) {
|
||||||
|
$this->output->writeln(' <fg=yellow;options=bold>WARN</> <fg=default>The [tests/.pest/shards.json] file is out of date. Run [--update-shards] to update it.</>');
|
||||||
|
}
|
||||||
|
|
||||||
return $exitCode;
|
return $exitCode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Terminates the plugin.
|
||||||
|
*/
|
||||||
|
public function terminate(): void
|
||||||
|
{
|
||||||
|
if (! self::$updateShards) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Parallel::isWorker()) {
|
||||||
|
$this->writeWorkerTimings();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! self::$passed) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$timings = self::$collectedTimings ?? $this->collectTimings();
|
||||||
|
|
||||||
|
if ($timings === []) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->writeTimings($timings);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Collects timings from subscribers or worker temp files.
|
||||||
|
*
|
||||||
|
* @return array<string, float>
|
||||||
|
*/
|
||||||
|
private function collectTimings(): array
|
||||||
|
{
|
||||||
|
$runId = Parallel::getGlobal('SHARD_RUN_ID');
|
||||||
|
|
||||||
|
if (is_string($runId)) {
|
||||||
|
return $this->readWorkerTimings($runId);
|
||||||
|
}
|
||||||
|
|
||||||
|
return EnsureShardTimingsAreCollected::timings();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes the current worker's timing data to a temp file.
|
||||||
|
*/
|
||||||
|
private function writeWorkerTimings(): void
|
||||||
|
{
|
||||||
|
$timings = EnsureShardTimingsAreCollected::timings();
|
||||||
|
|
||||||
|
if ($timings === []) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$runId = Parallel::getGlobal('SHARD_RUN_ID');
|
||||||
|
|
||||||
|
if (! is_string($runId)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$path = sys_get_temp_dir().DIRECTORY_SEPARATOR.'__pest_sharding_'.$runId.'-'.getmypid().'.json';
|
||||||
|
|
||||||
|
file_put_contents($path, json_encode($timings, JSON_THROW_ON_ERROR));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads and merges timing data from all worker temp files.
|
||||||
|
*
|
||||||
|
* @return array<string, float>
|
||||||
|
*/
|
||||||
|
private function readWorkerTimings(string $runId): array
|
||||||
|
{
|
||||||
|
$pattern = sys_get_temp_dir().DIRECTORY_SEPARATOR.'__pest_sharding_'.$runId.'-*.json';
|
||||||
|
$files = glob($pattern);
|
||||||
|
|
||||||
|
if ($files === false || $files === []) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$merged = [];
|
||||||
|
|
||||||
|
foreach ($files as $file) {
|
||||||
|
$contents = file_get_contents($file);
|
||||||
|
|
||||||
|
if ($contents === false) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$timings = json_decode($contents, true);
|
||||||
|
|
||||||
|
if (is_array($timings)) {
|
||||||
|
$merged = array_merge($merged, $timings);
|
||||||
|
}
|
||||||
|
|
||||||
|
unlink($file);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $merged;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the path to shards.json.
|
||||||
|
*/
|
||||||
|
private function shardsPath(): string
|
||||||
|
{
|
||||||
|
$testSuite = TestSuite::getInstance();
|
||||||
|
|
||||||
|
return implode(DIRECTORY_SEPARATOR, [$testSuite->rootPath, $testSuite->testPath, '.pest', 'shards.json']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads the timings from shards.json.
|
||||||
|
*
|
||||||
|
* @return array<string, float>|null
|
||||||
|
*/
|
||||||
|
private function loadShardsFile(): ?array
|
||||||
|
{
|
||||||
|
$path = $this->shardsPath();
|
||||||
|
|
||||||
|
if (! file_exists($path)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$contents = file_get_contents($path);
|
||||||
|
|
||||||
|
if ($contents === false) {
|
||||||
|
throw new InvalidOption('The [tests/.pest/shards.json] file could not be read. Delete it or run [--update-shards] to regenerate.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = json_decode($contents, true);
|
||||||
|
|
||||||
|
if (! is_array($data) || ! isset($data['timings']) || ! is_array($data['timings'])) {
|
||||||
|
throw new InvalidOption('The [tests/.pest/shards.json] file is corrupted. Delete it or run [--update-shards] to regenerate.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $data['timings'];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Partitions tests across shards using the LPT (Longest Processing Time) algorithm.
|
||||||
|
*
|
||||||
|
* @param list<string> $tests
|
||||||
|
* @param array<string, float> $timings
|
||||||
|
* @return list<list<string>>
|
||||||
|
*/
|
||||||
|
private function partitionByTime(array $tests, array $timings, int $total): array
|
||||||
|
{
|
||||||
|
$knownTimings = array_filter(
|
||||||
|
array_map(fn (string $test): ?float => $timings[$test] ?? null, $tests),
|
||||||
|
fn (?float $t): bool => $t !== null,
|
||||||
|
);
|
||||||
|
|
||||||
|
$median = $knownTimings !== [] ? $this->median(array_values($knownTimings)) : 1.0;
|
||||||
|
|
||||||
|
$testsWithTimings = array_map(
|
||||||
|
fn (string $test): array => ['test' => $test, 'time' => $timings[$test] ?? $median],
|
||||||
|
$tests,
|
||||||
|
);
|
||||||
|
|
||||||
|
usort($testsWithTimings, fn (array $a, array $b): int => $b['time'] <=> $a['time']);
|
||||||
|
|
||||||
|
/** @var list<list<string>> */
|
||||||
|
$bins = array_fill(0, $total, []);
|
||||||
|
/** @var non-empty-list<float> */
|
||||||
|
$binTimes = array_fill(0, $total, 0.0);
|
||||||
|
|
||||||
|
foreach ($testsWithTimings as $item) {
|
||||||
|
$minIndex = array_search(min($binTimes), $binTimes, strict: true);
|
||||||
|
assert(is_int($minIndex));
|
||||||
|
|
||||||
|
$bins[$minIndex][] = $item['test'];
|
||||||
|
$binTimes[$minIndex] += $item['time'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $bins;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calculates the median of an array of floats.
|
||||||
|
*
|
||||||
|
* @param list<float> $values
|
||||||
|
*/
|
||||||
|
private function median(array $values): float
|
||||||
|
{
|
||||||
|
sort($values);
|
||||||
|
|
||||||
|
$count = count($values);
|
||||||
|
$middle = (int) floor($count / 2);
|
||||||
|
|
||||||
|
if ($count % 2 === 0) {
|
||||||
|
return ($values[$middle - 1] + $values[$middle]) / 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $values[$middle];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes the timings to shards.json.
|
||||||
|
*
|
||||||
|
* @param array<string, float> $timings
|
||||||
|
*/
|
||||||
|
private function writeTimings(array $timings): void
|
||||||
|
{
|
||||||
|
$path = $this->shardsPath();
|
||||||
|
|
||||||
|
$directory = dirname($path);
|
||||||
|
if (! is_dir($directory)) {
|
||||||
|
mkdir($directory, 0755, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self::$knownTests !== null) {
|
||||||
|
$knownSet = array_flip(self::$knownTests);
|
||||||
|
$timings = array_intersect_key($timings, $knownSet);
|
||||||
|
}
|
||||||
|
|
||||||
|
ksort($timings);
|
||||||
|
|
||||||
|
$canonical = self::$knownTests ?? array_keys($timings);
|
||||||
|
sort($canonical);
|
||||||
|
|
||||||
|
file_put_contents($path, json_encode([
|
||||||
|
'timings' => $timings,
|
||||||
|
'checksum' => md5(implode("\n", $canonical)),
|
||||||
|
'updated_at' => date('c'),
|
||||||
|
], JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES)."\n");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the shard information.
|
* Returns the shard information.
|
||||||
*
|
*
|
||||||
|
|||||||
@ -5,7 +5,6 @@ declare(strict_types=1);
|
|||||||
namespace Pest\Plugins;
|
namespace Pest\Plugins;
|
||||||
|
|
||||||
use Pest\Contracts\Plugins\HandlesArguments;
|
use Pest\Contracts\Plugins\HandlesArguments;
|
||||||
use Pest\Exceptions\InvalidOption;
|
|
||||||
use Pest\TestSuite;
|
use Pest\TestSuite;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -15,21 +14,116 @@ final class Snapshot implements HandlesArguments
|
|||||||
{
|
{
|
||||||
use Concerns\HandleArguments;
|
use Concerns\HandleArguments;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether snapshots should be updated on this run.
|
||||||
|
*/
|
||||||
|
public static bool $updateSnapshots = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@inheritDoc}
|
* {@inheritDoc}
|
||||||
*/
|
*/
|
||||||
public function handleArguments(array $arguments): array
|
public function handleArguments(array $arguments): array
|
||||||
{
|
{
|
||||||
|
if (Parallel::isWorker() && Parallel::getGlobal('UPDATE_SNAPSHOTS') === true) {
|
||||||
|
self::$updateSnapshots = true;
|
||||||
|
|
||||||
|
return $arguments;
|
||||||
|
}
|
||||||
|
|
||||||
if (! $this->hasArgument('--update-snapshots', $arguments)) {
|
if (! $this->hasArgument('--update-snapshots', $arguments)) {
|
||||||
return $arguments;
|
return $arguments;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->hasArgument('--parallel', $arguments)) {
|
self::$updateSnapshots = true;
|
||||||
throw new InvalidOption('The [--update-snapshots] option is not supported when running in parallel.');
|
|
||||||
|
if ($this->isFullRun($arguments)) {
|
||||||
|
TestSuite::getInstance()->snapshots->flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
TestSuite::getInstance()->snapshots->flush();
|
if ($this->hasArgument('--parallel', $arguments) || $this->hasArgument('-p', $arguments)) {
|
||||||
|
Parallel::setGlobal('UPDATE_SNAPSHOTS', true);
|
||||||
|
}
|
||||||
|
|
||||||
return $this->popArgument('--update-snapshots', $arguments);
|
return $this->popArgument('--update-snapshots', $arguments);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Options that take a value as the next argument (rather than via "=value").
|
||||||
|
*
|
||||||
|
* @var list<string>
|
||||||
|
*/
|
||||||
|
private const array FLAGS_WITH_VALUES = [
|
||||||
|
'--filter',
|
||||||
|
'--group',
|
||||||
|
'--exclude-group',
|
||||||
|
'--test-suffix',
|
||||||
|
'--covers',
|
||||||
|
'--uses',
|
||||||
|
'--cache-directory',
|
||||||
|
'--cache-result-file',
|
||||||
|
'--configuration',
|
||||||
|
'--colors',
|
||||||
|
'--test-directory',
|
||||||
|
'--bootstrap',
|
||||||
|
'--order-by',
|
||||||
|
'--random-order-seed',
|
||||||
|
'--log-junit',
|
||||||
|
'--log-teamcity',
|
||||||
|
'--log-events-text',
|
||||||
|
'--log-events-verbose-text',
|
||||||
|
'--coverage-clover',
|
||||||
|
'--coverage-cobertura',
|
||||||
|
'--coverage-crap4j',
|
||||||
|
'--coverage-html',
|
||||||
|
'--coverage-php',
|
||||||
|
'--coverage-text',
|
||||||
|
'--coverage-xml',
|
||||||
|
'--assignee',
|
||||||
|
'--issue',
|
||||||
|
'--ticket',
|
||||||
|
'--pr',
|
||||||
|
'--pull-request',
|
||||||
|
'--retry',
|
||||||
|
'--shard',
|
||||||
|
'--repeat',
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines whether the command targets the entire suite (no filter, no path).
|
||||||
|
*
|
||||||
|
* @param array<int, string> $arguments
|
||||||
|
*/
|
||||||
|
private function isFullRun(array $arguments): bool
|
||||||
|
{
|
||||||
|
if ($this->hasArgument('--filter', $arguments)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tokens = array_slice($arguments, 1);
|
||||||
|
$skipNext = false;
|
||||||
|
|
||||||
|
foreach ($tokens as $arg) {
|
||||||
|
if ($skipNext) {
|
||||||
|
$skipNext = false;
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($arg === '') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($arg[0] === '-') {
|
||||||
|
if (in_array($arg, self::FLAGS_WITH_VALUES, true)) {
|
||||||
|
$skipNext = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
1780
src/Plugins/Tia.php
Normal file
1780
src/Plugins/Tia.php
Normal file
File diff suppressed because it is too large
Load Diff
621
src/Plugins/Tia/BaselineSync.php
Normal file
621
src/Plugins/Tia/BaselineSync.php
Normal file
@ -0,0 +1,621 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Exceptions\BaselineFetchFailed;
|
||||||
|
use Pest\Panic;
|
||||||
|
use Pest\Plugins\Tia;
|
||||||
|
use Pest\Plugins\Tia\Contracts\State;
|
||||||
|
use Pest\Support\View;
|
||||||
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
use Symfony\Component\Process\Process;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class BaselineSync
|
||||||
|
{
|
||||||
|
private const string WORKFLOW_FILE = 'tia-baseline.yml';
|
||||||
|
|
||||||
|
private const string ARTIFACT_NAME = 'pest-tia-baseline';
|
||||||
|
|
||||||
|
private const string GRAPH_ASSET = Tia::KEY_GRAPH;
|
||||||
|
|
||||||
|
private const string COVERAGE_ASSET = Tia::KEY_COVERAGE_CACHE;
|
||||||
|
|
||||||
|
private const string DOWNLOAD_CACHE_DIR = 'artifacts';
|
||||||
|
|
||||||
|
private const int DOWNLOAD_CACHE_MAX_ENTRIES = 5;
|
||||||
|
|
||||||
|
private const int FETCH_COOLDOWN_SECONDS = 86400;
|
||||||
|
|
||||||
|
private const array DIAGNOSES = [
|
||||||
|
'network' => [
|
||||||
|
'pattern' => '/could not resolve host|connection refused|connection reset|temporary failure in name resolution|network is unreachable|no route to host|i\/o timeout|tls handshake|getaddrinfo/i',
|
||||||
|
'message' => 'network error (offline or DNS unreachable). Try again when connected.',
|
||||||
|
],
|
||||||
|
'gh-auth' => [
|
||||||
|
'pattern' => '/authentication failed|not logged in|requires authentication|bad credentials|401/i',
|
||||||
|
'message' => 'authentication failed — run `gh auth login` and retry.',
|
||||||
|
],
|
||||||
|
'rate-limit' => [
|
||||||
|
'pattern' => '/rate limit|too many requests|secondary rate limit/i',
|
||||||
|
'message' => 'GitHub API rate limit hit — try again later.',
|
||||||
|
],
|
||||||
|
'not-found' => [
|
||||||
|
'pattern' => '/404|not found|repository not found/i',
|
||||||
|
'message' => 'workflow or artifact not found in repo.',
|
||||||
|
],
|
||||||
|
'forbidden' => [
|
||||||
|
'pattern' => '/403|forbidden|access denied/i',
|
||||||
|
'message' => 'access denied — check that your `gh` token has repo + actions read scope.',
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
private State $state,
|
||||||
|
private OutputInterface $output,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
private function renderBadge(string $type, string $content): void
|
||||||
|
{
|
||||||
|
View::render('components.badge', ['type' => $type, 'content' => $content]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function renderChild(string $text): void
|
||||||
|
{
|
||||||
|
$this->output->writeln(sprintf(' <fg=gray>─ %s</>', $text));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function fetchIfAvailable(string $projectRoot, bool $force = false, bool $hasAnchor = false): bool
|
||||||
|
{
|
||||||
|
$repo = $this->detectGitHubRepo($projectRoot);
|
||||||
|
|
||||||
|
if ($repo === null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $force && ($remaining = $this->cooldownRemaining()) !== null) {
|
||||||
|
$this->renderBadge('WARN', sprintf(
|
||||||
|
'Last fetch found no baseline — next auto-retry in %s. Override with --refetch.',
|
||||||
|
$this->formatDuration($remaining),
|
||||||
|
));
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$result = $this->download($repo, $projectRoot, $hasAnchor);
|
||||||
|
$payload = $result['payload'];
|
||||||
|
$failureKind = $result['failureKind'];
|
||||||
|
|
||||||
|
if ($payload === null) {
|
||||||
|
if ($failureKind === 'no-runs' || $failureKind === null) {
|
||||||
|
$this->startCooldown();
|
||||||
|
$this->emitPublishInstructions();
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $this->state->write(Tia::KEY_GRAPH, $payload['graph'])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($payload['coverage'] !== null) {
|
||||||
|
$this->state->write(Tia::KEY_COVERAGE_CACHE, $payload['coverage']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->clearCooldown();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function cooldownRemaining(): ?int
|
||||||
|
{
|
||||||
|
$raw = $this->state->read(Tia::KEY_FETCH_COOLDOWN);
|
||||||
|
|
||||||
|
if ($raw === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$decoded = json_decode($raw, true);
|
||||||
|
|
||||||
|
if (! is_array($decoded) || ! isset($decoded['until']) || ! is_int($decoded['until'])) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$remaining = $decoded['until'] - time();
|
||||||
|
|
||||||
|
return $remaining > 0 ? $remaining : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function startCooldown(): void
|
||||||
|
{
|
||||||
|
$this->state->write(Tia::KEY_FETCH_COOLDOWN, (string) json_encode([
|
||||||
|
'until' => time() + self::FETCH_COOLDOWN_SECONDS,
|
||||||
|
]));
|
||||||
|
}
|
||||||
|
|
||||||
|
private function clearCooldown(): void
|
||||||
|
{
|
||||||
|
$this->state->delete(Tia::KEY_FETCH_COOLDOWN);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function formatDuration(int $seconds): string
|
||||||
|
{
|
||||||
|
if ($seconds >= 3600) {
|
||||||
|
return (int) round($seconds / 3600).'h';
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($seconds >= 60) {
|
||||||
|
return (int) round($seconds / 60).'m';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $seconds.'s';
|
||||||
|
}
|
||||||
|
|
||||||
|
private function emitPublishInstructions(): void
|
||||||
|
{
|
||||||
|
if ($this->isCi()) {
|
||||||
|
$this->renderBadge('INFO', 'No baseline yet — this run will produce one.');
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->renderBadge('WARN', 'No baseline published yet — recording locally.');
|
||||||
|
$this->renderChild('See https://pestphp.com/docs/tia for how to publish one from CI.');
|
||||||
|
}
|
||||||
|
|
||||||
|
private function isCi(): bool
|
||||||
|
{
|
||||||
|
return getenv('GITHUB_ACTIONS') === 'true'
|
||||||
|
|| getenv('GITLAB_CI') === 'true'
|
||||||
|
|| getenv('CIRCLECI') === 'true';
|
||||||
|
}
|
||||||
|
|
||||||
|
private function detectGitHubRepo(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$gitConfig = $projectRoot.DIRECTORY_SEPARATOR.'.git'.DIRECTORY_SEPARATOR.'config';
|
||||||
|
|
||||||
|
if (! is_file($gitConfig)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$content = @file_get_contents($gitConfig);
|
||||||
|
|
||||||
|
if ($content === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (preg_match('/\[remote "origin"\][^\[]*?url\s*=\s*(\S+)/s', $content, $match) !== 1) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$url = $match[1];
|
||||||
|
|
||||||
|
if (preg_match('#^git@github\.com:([\w.-]+/[\w.-]+?)(?:\.git)?$#', $url, $m) === 1) {
|
||||||
|
return $m[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (preg_match('#^https?://github\.com/([\w.-]+/[\w.-]+?)(?:\.git)?/?$#', $url, $m) === 1) {
|
||||||
|
return $m[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (preg_match('#^ssh://(?:[^@/]+@)?github\.com(?::\d+)?/([\w.-]+/[\w.-]+?)(?:\.git)?/?$#i', $url, $m) === 1) {
|
||||||
|
return $m[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{payload: array{graph: string, coverage: ?string, sizeOnDisk: int}|null, failureKind: ?string}
|
||||||
|
*/
|
||||||
|
private function download(string $repo, string $projectRoot, bool $hasAnchor = false): array
|
||||||
|
{
|
||||||
|
$this->validateGhDependencies($hasAnchor);
|
||||||
|
|
||||||
|
[$runId, $listError] = $this->latestSuccessfulRunIdWithError($repo);
|
||||||
|
|
||||||
|
if ($listError !== null) {
|
||||||
|
$this->panicOnClassifiedError($listError, 'Failed to query baseline runs', $hasAnchor);
|
||||||
|
|
||||||
|
$this->renderBadge('WARN', sprintf(
|
||||||
|
'Failed to query baseline runs — %s',
|
||||||
|
$listError['message'],
|
||||||
|
));
|
||||||
|
|
||||||
|
return ['payload' => null, 'failureKind' => $listError['kind']];
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($runId === null) {
|
||||||
|
return ['payload' => null, 'failureKind' => 'no-runs'];
|
||||||
|
}
|
||||||
|
|
||||||
|
$runCacheDir = $this->downloadCacheDir($projectRoot).DIRECTORY_SEPARATOR.$this->safeRunId($runId);
|
||||||
|
|
||||||
|
if (is_file($runCacheDir.DIRECTORY_SEPARATOR.self::GRAPH_ASSET)) {
|
||||||
|
@touch($runCacheDir);
|
||||||
|
|
||||||
|
$this->renderChild(sprintf(
|
||||||
|
'Using cached baseline from %s (run %s).',
|
||||||
|
$repo,
|
||||||
|
$runId,
|
||||||
|
));
|
||||||
|
|
||||||
|
return ['payload' => $this->readArtifact($runCacheDir), 'failureKind' => null];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! @mkdir($runCacheDir, 0755, true) && ! is_dir($runCacheDir)) {
|
||||||
|
return ['payload' => null, 'failureKind' => null];
|
||||||
|
}
|
||||||
|
|
||||||
|
$download = $this->downloadArtifact($repo, $runId, $runCacheDir, $hasAnchor);
|
||||||
|
|
||||||
|
if (! $download['success']) {
|
||||||
|
return ['payload' => null, 'failureKind' => $download['failureKind']];
|
||||||
|
}
|
||||||
|
|
||||||
|
$payload = $this->validateDownloadedArtifact($runCacheDir, $hasAnchor);
|
||||||
|
|
||||||
|
$this->trimDownloadCache($projectRoot);
|
||||||
|
|
||||||
|
return ['payload' => $payload, 'failureKind' => null];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array{kind: string, message: string} $diagnosis
|
||||||
|
*/
|
||||||
|
private function panicOnClassifiedError(array $diagnosis, string $contextPrefix, bool $hasAnchor): void
|
||||||
|
{
|
||||||
|
if (! in_array($diagnosis['kind'], ['forbidden', 'not-found'], true)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Panic::with(new BaselineFetchFailed(
|
||||||
|
sprintf('%s — %s', $contextPrefix, $diagnosis['message']),
|
||||||
|
'Verify workflow tia-baseline.yml, artifact pest-tia-baseline, and gh token scope.',
|
||||||
|
$hasAnchor,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
private function validateGhDependencies(bool $hasAnchor): void
|
||||||
|
{
|
||||||
|
if (! $this->commandExists('gh')) {
|
||||||
|
Panic::with(new BaselineFetchFailed(
|
||||||
|
'GitHub CLI (gh) not found — cannot fetch baseline.',
|
||||||
|
'Install it from https://cli.github.com.',
|
||||||
|
$hasAnchor,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $this->ghAuthenticated()) {
|
||||||
|
Panic::with(new BaselineFetchFailed(
|
||||||
|
'GitHub CLI (gh) is not authenticated — cannot fetch baseline.',
|
||||||
|
'Run `gh auth login` and retry.',
|
||||||
|
$hasAnchor,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{success: bool, failureKind: ?string}
|
||||||
|
*/
|
||||||
|
private function downloadArtifact(string $repo, string $runId, string $runCacheDir, bool $hasAnchor): array
|
||||||
|
{
|
||||||
|
$artifactSize = $this->artifactSize($repo, $runId);
|
||||||
|
|
||||||
|
$this->output->writeln('');
|
||||||
|
$this->renderChild($artifactSize !== null
|
||||||
|
? sprintf(
|
||||||
|
'Downloading TIA baseline (%s) from %s…',
|
||||||
|
$this->formatSize($artifactSize),
|
||||||
|
$repo,
|
||||||
|
)
|
||||||
|
: sprintf(
|
||||||
|
'Downloading TIA baseline from %s…',
|
||||||
|
$repo,
|
||||||
|
));
|
||||||
|
|
||||||
|
$process = new Process([
|
||||||
|
'gh', 'run', 'download', $runId,
|
||||||
|
'-R', $repo,
|
||||||
|
'-n', self::ARTIFACT_NAME,
|
||||||
|
'-D', $runCacheDir,
|
||||||
|
]);
|
||||||
|
$process->setTimeout(900.0);
|
||||||
|
$process->start();
|
||||||
|
|
||||||
|
$startedAt = microtime(true);
|
||||||
|
$tick = 0;
|
||||||
|
|
||||||
|
while ($process->isRunning()) {
|
||||||
|
$this->renderDownloadProgress($startedAt, $tick++);
|
||||||
|
usleep(120_000);
|
||||||
|
}
|
||||||
|
|
||||||
|
$process->wait();
|
||||||
|
$this->clearProgressLine();
|
||||||
|
|
||||||
|
if ($process->isSuccessful()) {
|
||||||
|
return ['success' => true, 'failureKind' => null];
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->cleanup($runCacheDir);
|
||||||
|
|
||||||
|
$diagnosis = $this->classifyGhError($process->getErrorOutput().$process->getOutput());
|
||||||
|
|
||||||
|
$this->panicOnClassifiedError($diagnosis, 'Baseline download failed', $hasAnchor);
|
||||||
|
|
||||||
|
$this->renderBadge('WARN', sprintf(
|
||||||
|
'Baseline download failed — %s',
|
||||||
|
$diagnosis['message'],
|
||||||
|
));
|
||||||
|
|
||||||
|
return ['success' => false, 'failureKind' => $diagnosis['kind']];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{graph: string, coverage: ?string, sizeOnDisk: int}
|
||||||
|
*/
|
||||||
|
private function validateDownloadedArtifact(string $runCacheDir, bool $hasAnchor): array
|
||||||
|
{
|
||||||
|
$payload = $this->readArtifact($runCacheDir);
|
||||||
|
|
||||||
|
if ($payload === null) {
|
||||||
|
$this->cleanup($runCacheDir);
|
||||||
|
|
||||||
|
Panic::with(new BaselineFetchFailed(
|
||||||
|
'Baseline downloaded but the artifact is missing expected files (graph.json).',
|
||||||
|
'Your CI publish step is broken — check the workflow that uploads pest-tia-baseline.',
|
||||||
|
$hasAnchor,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $payload;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function artifactSize(string $repo, string $runId): ?int
|
||||||
|
{
|
||||||
|
$process = new Process([
|
||||||
|
'gh', 'api',
|
||||||
|
sprintf('repos/%s/actions/runs/%s/artifacts', $repo, $runId),
|
||||||
|
'--jq', sprintf(
|
||||||
|
'.artifacts[] | select(.name == "%s") | .size_in_bytes', // @pest-ignore-type
|
||||||
|
self::ARTIFACT_NAME,
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
$process->setTimeout(30.0);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$size = trim($process->getOutput());
|
||||||
|
|
||||||
|
return is_numeric($size) ? (int) $size : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function renderDownloadProgress(float $startedAt, int $tick): void
|
||||||
|
{
|
||||||
|
static $frames = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
|
||||||
|
|
||||||
|
$elapsed = max(0.0, microtime(true) - $startedAt);
|
||||||
|
$frame = $frames[$tick % count($frames)];
|
||||||
|
|
||||||
|
$this->output->write(sprintf(
|
||||||
|
"\r\033[K <fg=gray>%s %.1fs elapsed</>",
|
||||||
|
$frame,
|
||||||
|
$elapsed,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
private function clearProgressLine(): void
|
||||||
|
{
|
||||||
|
$this->output->write("\r\033[K");
|
||||||
|
}
|
||||||
|
|
||||||
|
private function dirSize(string $dir): int
|
||||||
|
{
|
||||||
|
if (! is_dir($dir)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$total = 0;
|
||||||
|
|
||||||
|
$iterator = new \RecursiveIteratorIterator(
|
||||||
|
new \RecursiveDirectoryIterator($dir, \FilesystemIterator::SKIP_DOTS),
|
||||||
|
);
|
||||||
|
|
||||||
|
/** @var \SplFileInfo $entry */
|
||||||
|
foreach ($iterator as $entry) {
|
||||||
|
if ($entry->isFile()) {
|
||||||
|
$total += $entry->getSize();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $total;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{graph: string, coverage: ?string, sizeOnDisk: int}|null
|
||||||
|
*/
|
||||||
|
private function readArtifact(string $dir): ?array
|
||||||
|
{
|
||||||
|
$graphPath = $dir.DIRECTORY_SEPARATOR.self::GRAPH_ASSET;
|
||||||
|
$coveragePath = $dir.DIRECTORY_SEPARATOR.self::COVERAGE_ASSET;
|
||||||
|
|
||||||
|
$graph = is_file($graphPath) ? @file_get_contents($graphPath) : false;
|
||||||
|
|
||||||
|
if ($graph === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$coverage = is_file($coveragePath) ? @file_get_contents($coveragePath) : false;
|
||||||
|
|
||||||
|
return [
|
||||||
|
'graph' => $graph,
|
||||||
|
'coverage' => $coverage === false ? null : $coverage,
|
||||||
|
'sizeOnDisk' => $this->dirSize($dir),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function downloadCacheDir(string $projectRoot): string
|
||||||
|
{
|
||||||
|
return Storage::tempDir($projectRoot).DIRECTORY_SEPARATOR.self::DOWNLOAD_CACHE_DIR;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function safeRunId(string $runId): string
|
||||||
|
{
|
||||||
|
$sanitised = preg_replace('/[^A-Za-z0-9_-]/', '', $runId) ?? '';
|
||||||
|
|
||||||
|
return $sanitised === '' ? 'unknown' : $sanitised;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function trimDownloadCache(string $projectRoot): void
|
||||||
|
{
|
||||||
|
$root = $this->downloadCacheDir($projectRoot);
|
||||||
|
|
||||||
|
if (! is_dir($root)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$entries = @scandir($root);
|
||||||
|
|
||||||
|
if ($entries === false) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$candidates = [];
|
||||||
|
|
||||||
|
foreach ($entries as $entry) {
|
||||||
|
if (in_array($entry, ['.', '..'], true)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$path = $root.DIRECTORY_SEPARATOR.$entry;
|
||||||
|
|
||||||
|
if (! is_dir($path)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$mtime = @filemtime($path);
|
||||||
|
$candidates[] = ['path' => $path, 'mtime' => $mtime === false ? 0 : $mtime];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count($candidates) <= self::DOWNLOAD_CACHE_MAX_ENTRIES) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
usort(
|
||||||
|
$candidates,
|
||||||
|
static fn (array $a, array $b): int => $b['mtime'] <=> $a['mtime'],
|
||||||
|
);
|
||||||
|
|
||||||
|
foreach (array_slice($candidates, self::DOWNLOAD_CACHE_MAX_ENTRIES) as $stale) {
|
||||||
|
$this->cleanup($stale['path']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{0: ?string, 1: ?array{kind: string, message: string}}
|
||||||
|
*/
|
||||||
|
private function latestSuccessfulRunIdWithError(string $repo): array
|
||||||
|
{
|
||||||
|
$process = new Process([
|
||||||
|
'gh', 'run', 'list',
|
||||||
|
'-R', $repo,
|
||||||
|
'--workflow', self::WORKFLOW_FILE,
|
||||||
|
'--status', 'success',
|
||||||
|
'--limit', '1',
|
||||||
|
'--json', 'databaseId',
|
||||||
|
'--jq', '.[0].databaseId // empty',
|
||||||
|
]);
|
||||||
|
$process->setTimeout(30.0);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
return [null, $this->classifyGhError($process->getErrorOutput().$process->getOutput())];
|
||||||
|
}
|
||||||
|
|
||||||
|
$runId = trim($process->getOutput());
|
||||||
|
|
||||||
|
return [$runId === '' ? null : $runId, null];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function ghAuthenticated(): bool
|
||||||
|
{
|
||||||
|
$process = new Process(['gh', 'auth', 'status']);
|
||||||
|
$process->setTimeout(10.0);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
return $process->isSuccessful();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{kind: string, message: string}
|
||||||
|
*/
|
||||||
|
private function classifyGhError(string $output): array
|
||||||
|
{
|
||||||
|
$output = trim($output);
|
||||||
|
|
||||||
|
if ($output === '') {
|
||||||
|
return ['kind' => 'unknown', 'message' => 'unknown error'];
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (self::DIAGNOSES as $kind => $diagnosis) {
|
||||||
|
if (preg_match($diagnosis['pattern'], $output) === 1) {
|
||||||
|
return ['kind' => $kind, 'message' => $diagnosis['message']];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ['kind' => 'unknown', 'message' => trim(strtok($output, "\n"))];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function commandExists(string $cmd): bool
|
||||||
|
{
|
||||||
|
$process = new Process(['which', $cmd]);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
return $process->isSuccessful();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function cleanup(string $dir): void
|
||||||
|
{
|
||||||
|
if (! is_dir($dir)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$iterator = new \RecursiveIteratorIterator(
|
||||||
|
new \RecursiveDirectoryIterator($dir, \FilesystemIterator::SKIP_DOTS),
|
||||||
|
\RecursiveIteratorIterator::CHILD_FIRST,
|
||||||
|
);
|
||||||
|
|
||||||
|
/** @var \SplFileInfo $entry */
|
||||||
|
foreach ($iterator as $entry) {
|
||||||
|
if ($entry->isDir()) {
|
||||||
|
@rmdir($entry->getPathname());
|
||||||
|
} else {
|
||||||
|
@unlink($entry->getPathname());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@rmdir($dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function formatSize(int $bytes): string
|
||||||
|
{
|
||||||
|
if ($bytes >= 1024 * 1024) {
|
||||||
|
return sprintf('%.1f MB', $bytes / 1024 / 1024);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($bytes >= 1024) {
|
||||||
|
return sprintf('%.1f KB', $bytes / 1024);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $bytes.' B';
|
||||||
|
}
|
||||||
|
}
|
||||||
28
src/Plugins/Tia/Bootstrapper.php
Normal file
28
src/Plugins/Tia/Bootstrapper.php
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Contracts\Bootstrapper as BootstrapperContract;
|
||||||
|
use Pest\Plugins\Tia\Contracts\State;
|
||||||
|
use Pest\Support\Container;
|
||||||
|
use Pest\TestSuite;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Bootstrapper implements BootstrapperContract
|
||||||
|
{
|
||||||
|
public function __construct(private Container $container) {}
|
||||||
|
|
||||||
|
public function boot(): void
|
||||||
|
{
|
||||||
|
$testSuite = $this->container->get(TestSuite::class);
|
||||||
|
assert($testSuite instanceof TestSuite);
|
||||||
|
|
||||||
|
$tempDir = Storage::tempDir($testSuite->rootPath);
|
||||||
|
|
||||||
|
$this->container->add(State::class, new FileState($tempDir));
|
||||||
|
}
|
||||||
|
}
|
||||||
326
src/Plugins/Tia/ChangedFiles.php
Normal file
326
src/Plugins/Tia/ChangedFiles.php
Normal file
@ -0,0 +1,326 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Exceptions\MissingDependency;
|
||||||
|
use Symfony\Component\Process\Process;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class ChangedFiles
|
||||||
|
{
|
||||||
|
public function __construct(private string $projectRoot) {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $files project-relative paths.
|
||||||
|
* @param array<string, string> $lastRunTree path → content hash from last run.
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
public function filterUnchangedSinceLastRun(array $files, array $lastRunTree): array
|
||||||
|
{
|
||||||
|
if ($lastRunTree === []) {
|
||||||
|
return $files;
|
||||||
|
}
|
||||||
|
|
||||||
|
$candidates = array_fill_keys($files, true);
|
||||||
|
|
||||||
|
foreach (array_keys($lastRunTree) as $snapshotted) {
|
||||||
|
$candidates[$snapshotted] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$remaining = [];
|
||||||
|
|
||||||
|
foreach (array_keys($candidates) as $file) {
|
||||||
|
$snapshot = $lastRunTree[$file] ?? null;
|
||||||
|
$current = $this->currentHash($file);
|
||||||
|
|
||||||
|
if ($snapshot === null || $current === null || $current !== $snapshot) {
|
||||||
|
$remaining[] = $file;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $remaining;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function currentHash(string $relativePath): ?string
|
||||||
|
{
|
||||||
|
$absolute = $this->projectRoot.DIRECTORY_SEPARATOR.$relativePath;
|
||||||
|
|
||||||
|
if (! is_file($absolute)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$hash = ContentHash::of($absolute);
|
||||||
|
|
||||||
|
return $hash === false ? null : $hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $files
|
||||||
|
* @return array<string, string> path → xxh128 content hash
|
||||||
|
*/
|
||||||
|
public function snapshotTree(array $files): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($files as $file) {
|
||||||
|
$absolute = $this->projectRoot.DIRECTORY_SEPARATOR.$file;
|
||||||
|
|
||||||
|
if (! is_file($absolute)) {
|
||||||
|
$out[$file] = '';
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$hash = ContentHash::of($absolute);
|
||||||
|
|
||||||
|
if ($hash !== false) {
|
||||||
|
$out[$file] = $hash;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>|null `null` when git is unavailable, or when
|
||||||
|
*/
|
||||||
|
public function since(?string $sha): ?array
|
||||||
|
{
|
||||||
|
$files = [];
|
||||||
|
|
||||||
|
if ($sha !== null && $sha !== '') {
|
||||||
|
if (! $this->shaIsReachable($sha)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$files = array_merge($files, $this->diffSinceSha($sha));
|
||||||
|
}
|
||||||
|
|
||||||
|
$files = array_merge($files, $this->workingTreeChanges());
|
||||||
|
|
||||||
|
$unique = [];
|
||||||
|
|
||||||
|
foreach ($files as $file) {
|
||||||
|
if ($file === '') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$unique[$file] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$candidates = array_keys($this->filterIgnored($unique));
|
||||||
|
|
||||||
|
if ($sha !== null && $sha !== '') {
|
||||||
|
return $this->filterBehaviourallyUnchanged($candidates, $sha);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $candidates;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $files
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private function filterBehaviourallyUnchanged(array $files, string $sha): array
|
||||||
|
{
|
||||||
|
$remaining = [];
|
||||||
|
|
||||||
|
foreach ($files as $file) {
|
||||||
|
$currentHash = $this->currentHash($file);
|
||||||
|
|
||||||
|
if ($currentHash === null) {
|
||||||
|
$remaining[] = $file;
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$baselineContent = $this->contentAtSha($sha, $file);
|
||||||
|
|
||||||
|
if ($baselineContent === null) {
|
||||||
|
$remaining[] = $file;
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($currentHash !== ContentHash::ofContent($file, $baselineContent)) {
|
||||||
|
$remaining[] = $file;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $remaining;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function contentAtSha(string $sha, string $path): ?string
|
||||||
|
{
|
||||||
|
$process = new Process(['git', 'show', $sha.':'.$path], $this->projectRoot);
|
||||||
|
$process->setTimeout(5.0);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $process->getOutput();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, true> $candidates
|
||||||
|
* @return array<string, true>
|
||||||
|
*/
|
||||||
|
private function filterIgnored(array $candidates): array
|
||||||
|
{
|
||||||
|
if ($candidates === []) {
|
||||||
|
return $candidates;
|
||||||
|
}
|
||||||
|
|
||||||
|
$process = new Process(
|
||||||
|
['git', 'check-ignore', '--no-index', '-z', '--stdin'],
|
||||||
|
$this->projectRoot,
|
||||||
|
);
|
||||||
|
$process->setTimeout(5.0);
|
||||||
|
$process->setInput(implode("\x00", array_keys($candidates)));
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
$exitCode = $process->getExitCode();
|
||||||
|
|
||||||
|
if ($exitCode !== 0 && $exitCode !== 1) {
|
||||||
|
throw new MissingDependency('Tia mode', 'git');
|
||||||
|
}
|
||||||
|
|
||||||
|
$output = $process->getOutput();
|
||||||
|
|
||||||
|
if ($output === '') {
|
||||||
|
return $candidates;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (explode("\x00", rtrim($output, "\x00")) as $ignored) {
|
||||||
|
if ($ignored !== '') {
|
||||||
|
unset($candidates[$ignored]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $candidates;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function currentBranch(): ?string
|
||||||
|
{
|
||||||
|
$process = new Process(['git', 'rev-parse', '--abbrev-ref', 'HEAD'], $this->projectRoot);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
throw new MissingDependency('Tia mode', 'git');
|
||||||
|
}
|
||||||
|
|
||||||
|
$branch = trim($process->getOutput());
|
||||||
|
|
||||||
|
return $branch === '' || $branch === 'HEAD' ? null : $branch;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function shaIsReachable(string $sha): bool
|
||||||
|
{
|
||||||
|
$process = new Process(
|
||||||
|
['git', 'merge-base', '--is-ancestor', $sha, 'HEAD'],
|
||||||
|
$this->projectRoot,
|
||||||
|
);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
return $process->getExitCode() === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private function diffSinceSha(string $sha): array
|
||||||
|
{
|
||||||
|
$process = new Process(
|
||||||
|
['git', 'diff', '--name-only', $sha.'..HEAD'],
|
||||||
|
$this->projectRoot,
|
||||||
|
);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
throw new MissingDependency('Tia mode', 'git');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->splitLines($process->getOutput());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private function workingTreeChanges(): array
|
||||||
|
{
|
||||||
|
$process = new Process(
|
||||||
|
['git', 'status', '--porcelain', '-z', '--untracked-files=all'],
|
||||||
|
$this->projectRoot,
|
||||||
|
);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
throw new MissingDependency('Tia mode', 'git');
|
||||||
|
}
|
||||||
|
|
||||||
|
$output = $process->getOutput();
|
||||||
|
|
||||||
|
if ($output === '') {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$records = explode("\x00", rtrim($output, "\x00"));
|
||||||
|
$files = [];
|
||||||
|
$count = count($records);
|
||||||
|
|
||||||
|
for ($i = 0; $i < $count; $i++) {
|
||||||
|
$record = $records[$i];
|
||||||
|
|
||||||
|
if (strlen($record) < 4) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$status = substr($record, 0, 2);
|
||||||
|
$path = substr($record, 3);
|
||||||
|
|
||||||
|
if ($status[0] === 'R' || $status[0] === 'C') {
|
||||||
|
$files[] = $path;
|
||||||
|
|
||||||
|
if (isset($records[$i + 1]) && $records[$i + 1] !== '') {
|
||||||
|
$files[] = $records[$i + 1];
|
||||||
|
$i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$files[] = $path;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $files;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function currentSha(): ?string
|
||||||
|
{
|
||||||
|
$process = new Process(['git', 'rev-parse', 'HEAD'], $this->projectRoot);
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
throw new MissingDependency('Tia mode', 'git');
|
||||||
|
}
|
||||||
|
|
||||||
|
$sha = trim($process->getOutput());
|
||||||
|
|
||||||
|
return $sha === '' ? null : $sha;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private function splitLines(string $output): array
|
||||||
|
{
|
||||||
|
$lines = preg_split('/\R+/', trim($output), flags: PREG_SPLIT_NO_EMPTY);
|
||||||
|
|
||||||
|
return $lines === false ? [] : $lines;
|
||||||
|
}
|
||||||
|
}
|
||||||
28
src/Plugins/Tia/Collectors.php
Normal file
28
src/Plugins/Tia/Collectors.php
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Edges\BladeEdges;
|
||||||
|
use Pest\Plugins\Tia\Edges\InertiaEdges;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class Collectors
|
||||||
|
{
|
||||||
|
/** @var list<class-string> */
|
||||||
|
private const array COLLECTORS = [
|
||||||
|
BladeEdges::class,
|
||||||
|
TableTracker::class,
|
||||||
|
InertiaEdges::class,
|
||||||
|
];
|
||||||
|
|
||||||
|
public static function armAll(Recorder $recorder): void
|
||||||
|
{
|
||||||
|
foreach (self::COLLECTORS as $collector) {
|
||||||
|
$collector::arm($recorder);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
75
src/Plugins/Tia/Configuration.php
Normal file
75
src/Plugins/Tia/Configuration.php
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Support\Container;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class Configuration
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @return $this
|
||||||
|
*/
|
||||||
|
public function always(): self
|
||||||
|
{
|
||||||
|
/** @var WatchPatterns $watchPatterns */
|
||||||
|
$watchPatterns = Container::getInstance()->get(WatchPatterns::class);
|
||||||
|
$watchPatterns->markEnabled();
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return $this
|
||||||
|
*/
|
||||||
|
public function locally(): self
|
||||||
|
{
|
||||||
|
/** @var WatchPatterns $watchPatterns */
|
||||||
|
$watchPatterns = Container::getInstance()->get(WatchPatterns::class);
|
||||||
|
$watchPatterns->markEnabled();
|
||||||
|
$watchPatterns->markLocally();
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return $this
|
||||||
|
*/
|
||||||
|
public function filtered(): self
|
||||||
|
{
|
||||||
|
/** @var WatchPatterns $watchPatterns */
|
||||||
|
$watchPatterns = Container::getInstance()->get(WatchPatterns::class);
|
||||||
|
$watchPatterns->markFiltered();
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return $this
|
||||||
|
*/
|
||||||
|
public function baselined(): self
|
||||||
|
{
|
||||||
|
/** @var WatchPatterns $watchPatterns */
|
||||||
|
$watchPatterns = Container::getInstance()->get(WatchPatterns::class);
|
||||||
|
$watchPatterns->markBaselined();
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, string> $patterns glob → project-relative test dir
|
||||||
|
* @return $this
|
||||||
|
*/
|
||||||
|
public function watch(array $patterns): self
|
||||||
|
{
|
||||||
|
/** @var WatchPatterns $watchPatterns */
|
||||||
|
$watchPatterns = Container::getInstance()->get(WatchPatterns::class);
|
||||||
|
$watchPatterns->add($patterns);
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
}
|
||||||
90
src/Plugins/Tia/ContentHash.php
Normal file
90
src/Plugins/Tia/ContentHash.php
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class ContentHash
|
||||||
|
{
|
||||||
|
public static function of(string $absolute): string|false
|
||||||
|
{
|
||||||
|
$raw = @file_get_contents($absolute);
|
||||||
|
|
||||||
|
if ($raw === false) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::ofContent($absolute, $raw);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function ofContent(string $path, string $raw): string
|
||||||
|
{
|
||||||
|
$lower = strtolower($path);
|
||||||
|
|
||||||
|
if (str_ends_with($lower, '.blade.php')) {
|
||||||
|
return self::hashBladeContent($raw);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (str_ends_with($lower, '.php')) {
|
||||||
|
return self::hashPhpContent($raw);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (['.vue', '.tsx', '.jsx', '.svelte', '.ts', '.js', '.mjs', '.cjs', '.mts'] as $extension) {
|
||||||
|
if (str_ends_with($lower, $extension)) {
|
||||||
|
return self::hashJsContent($raw);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return hash('xxh128', $raw);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function hashPhpContent(string $raw): string
|
||||||
|
{
|
||||||
|
$tokens = @token_get_all($raw);
|
||||||
|
|
||||||
|
if ($tokens === []) {
|
||||||
|
return hash('xxh128', $raw);
|
||||||
|
}
|
||||||
|
|
||||||
|
$normalised = '';
|
||||||
|
|
||||||
|
foreach ($tokens as $token) {
|
||||||
|
if (is_array($token)) {
|
||||||
|
if ($token[0] === T_WHITESPACE) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($token[0] === T_COMMENT) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($token[0] === T_DOC_COMMENT) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$normalised .= $token[1];
|
||||||
|
} else {
|
||||||
|
$normalised .= $token;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return hash('xxh128', $normalised);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function hashBladeContent(string $raw): string
|
||||||
|
{
|
||||||
|
$stripped = preg_replace('/\{\{--.*?--\}\}/s', '', $raw) ?? $raw;
|
||||||
|
$stripped = preg_replace('/\s+/', ' ', $stripped) ?? $stripped;
|
||||||
|
|
||||||
|
return hash('xxh128', trim($stripped));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function hashJsContent(string $raw): string
|
||||||
|
{
|
||||||
|
$stripped = preg_replace('/^\s*\/\/[^\n]*$/m', '', $raw) ?? $raw;
|
||||||
|
$stripped = preg_replace('/^\s*\/\*.*?\*\/\s*$/sm', '', $stripped) ?? $stripped;
|
||||||
|
$stripped = preg_replace('/\s+/', ' ', $stripped) ?? $stripped;
|
||||||
|
|
||||||
|
return hash('xxh128', trim($stripped));
|
||||||
|
}
|
||||||
|
}
|
||||||
24
src/Plugins/Tia/Contracts/State.php
Normal file
24
src/Plugins/Tia/Contracts/State.php
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\Contracts;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
interface State
|
||||||
|
{
|
||||||
|
public function read(string $key): ?string;
|
||||||
|
|
||||||
|
public function write(string $key, string $content): bool;
|
||||||
|
|
||||||
|
public function delete(string $key): bool;
|
||||||
|
|
||||||
|
public function exists(string $key): bool;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
public function keysWithPrefix(string $prefix): array;
|
||||||
|
}
|
||||||
18
src/Plugins/Tia/Contracts/WatchDefault.php
Normal file
18
src/Plugins/Tia/Contracts/WatchDefault.php
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\Contracts;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
interface WatchDefault
|
||||||
|
{
|
||||||
|
public function applicable(): bool;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, array<int, string>> pattern → list of project-relative test dirs
|
||||||
|
*/
|
||||||
|
public function defaults(string $projectRoot, string $testPath): array;
|
||||||
|
}
|
||||||
110
src/Plugins/Tia/CoverageCollector.php
Normal file
110
src/Plugins/Tia/CoverageCollector.php
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use PHPUnit\Runner\CodeCoverage as PhpUnitCodeCoverage;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class CoverageCollector
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var array<string, string|null>
|
||||||
|
*/
|
||||||
|
private array $classFileCache = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, array<int, string>>
|
||||||
|
*/
|
||||||
|
public function perTestFiles(): array
|
||||||
|
{
|
||||||
|
if (! PhpUnitCodeCoverage::instance()->isActive()) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$lineCoverage = PhpUnitCodeCoverage::instance()
|
||||||
|
->codeCoverage()
|
||||||
|
->getData()
|
||||||
|
->lineCoverage();
|
||||||
|
} catch (Throwable) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var array<string, array<string, true>> $edges */
|
||||||
|
$edges = [];
|
||||||
|
|
||||||
|
foreach ($lineCoverage as $sourceFile => $lines) {
|
||||||
|
$testIds = [];
|
||||||
|
|
||||||
|
foreach ($lines as $hits) {
|
||||||
|
if ($hits === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($hits as $id) {
|
||||||
|
$testIds[$id] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (array_keys($testIds) as $testId) {
|
||||||
|
$testFile = $this->testIdToFile($testId);
|
||||||
|
|
||||||
|
if ($testFile === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$edges[$testFile][$sourceFile] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($edges as $testFile => $sources) {
|
||||||
|
$out[$testFile] = array_keys($sources);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reset(): void
|
||||||
|
{
|
||||||
|
$this->classFileCache = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function testIdToFile(string $testId): ?string
|
||||||
|
{
|
||||||
|
$hash = strpos($testId, '#');
|
||||||
|
$identifier = $hash === false ? $testId : substr($testId, 0, $hash);
|
||||||
|
|
||||||
|
if (! str_contains($identifier, '::')) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
[$className] = explode('::', $identifier, 2);
|
||||||
|
|
||||||
|
if (array_key_exists($className, $this->classFileCache)) {
|
||||||
|
return $this->classFileCache[$className];
|
||||||
|
}
|
||||||
|
|
||||||
|
$file = $this->resolveClassFile($className);
|
||||||
|
$this->classFileCache[$className] = $file;
|
||||||
|
|
||||||
|
return $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function resolveClassFile(string $className): ?string
|
||||||
|
{
|
||||||
|
if (! class_exists($className, false)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(property_exists($className, '__filename') && is_string($className::$__filename));
|
||||||
|
|
||||||
|
return $className::$__filename;
|
||||||
|
}
|
||||||
|
}
|
||||||
177
src/Plugins/Tia/CoverageMerger.php
Normal file
177
src/Plugins/Tia/CoverageMerger.php
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia;
|
||||||
|
use Pest\Plugins\Tia\Contracts\State;
|
||||||
|
use Pest\Support\Container;
|
||||||
|
use SebastianBergmann\CodeCoverage\CodeCoverage;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class CoverageMerger
|
||||||
|
{
|
||||||
|
public static function applyIfMarked(string $reportPath): void
|
||||||
|
{
|
||||||
|
$state = self::state();
|
||||||
|
|
||||||
|
if (! $state->exists(Tia::KEY_COVERAGE_MARKER)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$state->delete(Tia::KEY_COVERAGE_MARKER);
|
||||||
|
|
||||||
|
$cachedBytes = $state->read(Tia::KEY_COVERAGE_CACHE);
|
||||||
|
|
||||||
|
if ($cachedBytes === null) {
|
||||||
|
$current = self::requireCoverage($reportPath);
|
||||||
|
|
||||||
|
if ($current instanceof CodeCoverage) {
|
||||||
|
self::primeUncoveredFiles($current);
|
||||||
|
$state->write(Tia::KEY_COVERAGE_CACHE, self::compress(serialize($current)));
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$decoded = self::decompress($cachedBytes);
|
||||||
|
|
||||||
|
if ($decoded === null) {
|
||||||
|
$state->delete(Tia::KEY_COVERAGE_CACHE);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$cached = self::unserializeCoverage($decoded);
|
||||||
|
$current = self::requireCoverage($reportPath);
|
||||||
|
|
||||||
|
if (! $cached instanceof CodeCoverage || ! $current instanceof CodeCoverage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
self::primeUncoveredFiles($cached);
|
||||||
|
self::primeUncoveredFiles($current);
|
||||||
|
|
||||||
|
self::stripCurrentTestsFromCached($cached, $current);
|
||||||
|
|
||||||
|
$cached->merge($current);
|
||||||
|
|
||||||
|
$serialised = serialize($cached);
|
||||||
|
|
||||||
|
@file_put_contents(
|
||||||
|
$reportPath,
|
||||||
|
'<?php return unserialize('.var_export($serialised, true).");\n",
|
||||||
|
);
|
||||||
|
$state->write(Tia::KEY_COVERAGE_CACHE, self::compress($serialised));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function primeUncoveredFiles(CodeCoverage $coverage): void
|
||||||
|
{
|
||||||
|
$coverage->getData(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function compress(string $bytes): string
|
||||||
|
{
|
||||||
|
$compressed = @gzencode($bytes);
|
||||||
|
|
||||||
|
return $compressed === false ? $bytes : $compressed;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function decompress(string $bytes): ?string
|
||||||
|
{
|
||||||
|
$decoded = @gzdecode($bytes);
|
||||||
|
|
||||||
|
return $decoded === false ? null : $decoded;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function stripCurrentTestsFromCached(CodeCoverage $cached, CodeCoverage $current): void
|
||||||
|
{
|
||||||
|
$currentIds = self::collectTestIds($current);
|
||||||
|
|
||||||
|
if ($currentIds === []) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$cachedData = $cached->getData();
|
||||||
|
$lineCoverage = $cachedData->lineCoverage();
|
||||||
|
|
||||||
|
foreach ($lineCoverage as $file => $lines) {
|
||||||
|
foreach ($lines as $line => $ids) {
|
||||||
|
if ($ids === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($ids === []) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$filtered = array_values(array_diff($ids, $currentIds));
|
||||||
|
|
||||||
|
if ($filtered !== $ids) {
|
||||||
|
$lineCoverage[$file][$line] = $filtered;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$cachedData->setLineCoverage($lineCoverage);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
private static function collectTestIds(CodeCoverage $coverage): array
|
||||||
|
{
|
||||||
|
$ids = [];
|
||||||
|
|
||||||
|
foreach ($coverage->getData()->lineCoverage() as $lines) {
|
||||||
|
foreach ($lines as $hits) {
|
||||||
|
if ($hits === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($hits as $id) {
|
||||||
|
$ids[$id] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_keys($ids);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function state(): State
|
||||||
|
{
|
||||||
|
$state = Container::getInstance()->get(State::class);
|
||||||
|
assert($state instanceof State);
|
||||||
|
|
||||||
|
return $state;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function requireCoverage(string $reportPath): ?CodeCoverage
|
||||||
|
{
|
||||||
|
if (! is_file($reportPath)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
/** @var mixed $value */
|
||||||
|
$value = require $reportPath;
|
||||||
|
} catch (Throwable) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $value instanceof CodeCoverage ? $value : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function unserializeCoverage(string $bytes): ?CodeCoverage
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$value = @unserialize($bytes);
|
||||||
|
} catch (Throwable) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $value instanceof CodeCoverage ? $value : null;
|
||||||
|
}
|
||||||
|
}
|
||||||
62
src/Plugins/Tia/Edges/BladeEdges.php
Normal file
62
src/Plugins/Tia/Edges/BladeEdges.php
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\Edges;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Recorder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class BladeEdges
|
||||||
|
{
|
||||||
|
private const string CONTAINER_CLASS = '\\Illuminate\\Container\\Container';
|
||||||
|
|
||||||
|
private const string MARKER = 'pest.tia.blade-edges-armed';
|
||||||
|
|
||||||
|
public static function arm(Recorder $recorder): void
|
||||||
|
{
|
||||||
|
if (! $recorder->isActive()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$containerClass = self::CONTAINER_CLASS;
|
||||||
|
|
||||||
|
if (! class_exists($containerClass)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var object $app */
|
||||||
|
$app = $containerClass::getInstance();
|
||||||
|
|
||||||
|
if (! method_exists($app, 'bound') || ! method_exists($app, 'make') || ! method_exists($app, 'instance')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($app->bound(self::MARKER) || ! $app->bound('view')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$app->instance(self::MARKER, true);
|
||||||
|
|
||||||
|
$factory = $app->make('view');
|
||||||
|
|
||||||
|
if (! is_object($factory) || ! method_exists($factory, 'composer')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$factory->composer('*', static function (object $view) use ($recorder): void {
|
||||||
|
if (! method_exists($view, 'getPath')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var mixed $path */
|
||||||
|
$path = $view->getPath();
|
||||||
|
|
||||||
|
if (is_string($path) && $path !== '') {
|
||||||
|
$recorder->linkSource($path);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
131
src/Plugins/Tia/Edges/InertiaEdges.php
Normal file
131
src/Plugins/Tia/Edges/InertiaEdges.php
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\Edges;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Recorder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class InertiaEdges
|
||||||
|
{
|
||||||
|
private const string CONTAINER_CLASS = '\\Illuminate\\Container\\Container';
|
||||||
|
|
||||||
|
private const string REQUEST_HANDLED_EVENT = 'Illuminate\\Foundation\\Http\\Events\\RequestHandled';
|
||||||
|
|
||||||
|
private const string MARKER = 'pest.tia.inertia-edges-armed';
|
||||||
|
|
||||||
|
public static function arm(Recorder $recorder): void
|
||||||
|
{
|
||||||
|
if (! $recorder->isActive()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$containerClass = self::CONTAINER_CLASS;
|
||||||
|
|
||||||
|
if (! class_exists($containerClass)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var object $app */
|
||||||
|
$app = $containerClass::getInstance();
|
||||||
|
|
||||||
|
if (! method_exists($app, 'bound') || ! method_exists($app, 'make') || ! method_exists($app, 'instance')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($app->bound(self::MARKER) || ! $app->bound('events')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$app->instance(self::MARKER, true);
|
||||||
|
|
||||||
|
/** @var object $events */
|
||||||
|
$events = $app->make('events');
|
||||||
|
|
||||||
|
if (! method_exists($events, 'listen')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$events->listen(self::REQUEST_HANDLED_EVENT, static function (object $event) use ($recorder): void {
|
||||||
|
if (! property_exists($event, 'response') || ! is_object($event->response)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$component = self::extractComponent($event->response);
|
||||||
|
|
||||||
|
if ($component !== null) {
|
||||||
|
$recorder->linkInertiaComponent($component);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function extractComponent(object $response): ?string
|
||||||
|
{
|
||||||
|
$content = self::readContent($response);
|
||||||
|
|
||||||
|
if ($content === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self::isInertiaJsonResponse($response)) {
|
||||||
|
return self::componentFromJson($content);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (str_contains($content, 'type="application/json"')
|
||||||
|
&& preg_match('#<script\b(?=[^>]*\bdata-page="app")(?=[^>]*\btype="application/json")[^>]*>(.+?)</script>#s', $content, $match) === 1) {
|
||||||
|
$component = self::componentFromJson(html_entity_decode($match[1]));
|
||||||
|
|
||||||
|
if ($component !== null) {
|
||||||
|
return $component;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (str_contains($content, 'data-page=')
|
||||||
|
&& preg_match('/\sdata-page="(\{[^"]+\})"/', $content, $match) === 1) {
|
||||||
|
return self::componentFromJson(html_entity_decode($match[1]));
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function isInertiaJsonResponse(object $response): bool
|
||||||
|
{
|
||||||
|
if (! property_exists($response, 'headers') || ! is_object($response->headers)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$headers = $response->headers;
|
||||||
|
|
||||||
|
return method_exists($headers, 'has') && $headers->has('X-Inertia') === true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function componentFromJson(string $json): ?string
|
||||||
|
{
|
||||||
|
/** @var mixed $decoded */
|
||||||
|
$decoded = json_decode($json, true);
|
||||||
|
|
||||||
|
if (is_array($decoded)
|
||||||
|
&& isset($decoded['component'])
|
||||||
|
&& is_string($decoded['component'])
|
||||||
|
&& $decoded['component'] !== '') {
|
||||||
|
return $decoded['component'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function readContent(object $response): ?string
|
||||||
|
{
|
||||||
|
if (! method_exists($response, 'getContent')) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var mixed $content */
|
||||||
|
$content = $response->getContent();
|
||||||
|
|
||||||
|
return is_string($content) ? $content : null;
|
||||||
|
}
|
||||||
|
}
|
||||||
35
src/Plugins/Tia/Enums/ReplayType.php
Normal file
35
src/Plugins/Tia/Enums/ReplayType.php
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\Enums;
|
||||||
|
|
||||||
|
use PHPUnit\Framework\TestStatus\TestStatus;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
enum ReplayType
|
||||||
|
{
|
||||||
|
case None;
|
||||||
|
case Pass;
|
||||||
|
case Risky;
|
||||||
|
case Skipped;
|
||||||
|
case Incomplete;
|
||||||
|
case Failure;
|
||||||
|
|
||||||
|
public static function fromStatus(?TestStatus $status): self
|
||||||
|
{
|
||||||
|
if (! $status instanceof TestStatus) {
|
||||||
|
return self::None;
|
||||||
|
}
|
||||||
|
|
||||||
|
return match (true) {
|
||||||
|
$status->isSuccess() => self::Pass,
|
||||||
|
$status->isRisky() => self::Risky,
|
||||||
|
$status->isSkipped() => self::Skipped,
|
||||||
|
$status->isIncomplete() => self::Incomplete,
|
||||||
|
default => self::Failure,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
130
src/Plugins/Tia/FileState.php
Normal file
130
src/Plugins/Tia/FileState.php
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Contracts\State;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class FileState implements State
|
||||||
|
{
|
||||||
|
private readonly string $rootDir;
|
||||||
|
|
||||||
|
private ?string $resolvedRoot = null;
|
||||||
|
|
||||||
|
public function __construct(string $rootDir)
|
||||||
|
{
|
||||||
|
$this->rootDir = rtrim($rootDir, DIRECTORY_SEPARATOR);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function read(string $key): ?string
|
||||||
|
{
|
||||||
|
$path = $this->pathFor($key);
|
||||||
|
|
||||||
|
if (! is_file($path)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$bytes = @file_get_contents($path);
|
||||||
|
|
||||||
|
return $bytes === false ? null : $bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function write(string $key, string $content): bool
|
||||||
|
{
|
||||||
|
if (! $this->ensureRoot()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$path = $this->pathFor($key);
|
||||||
|
$tmp = $path.'.'.bin2hex(random_bytes(4)).'.tmp';
|
||||||
|
|
||||||
|
if (@file_put_contents($tmp, $content) === false) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! @rename($tmp, $path)) {
|
||||||
|
@unlink($tmp);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function delete(string $key): bool
|
||||||
|
{
|
||||||
|
$path = $this->pathFor($key);
|
||||||
|
|
||||||
|
if (! is_file($path)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return @unlink($path);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function exists(string $key): bool
|
||||||
|
{
|
||||||
|
return is_file($this->pathFor($key));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function keysWithPrefix(string $prefix): array
|
||||||
|
{
|
||||||
|
$root = $this->resolvedRoot();
|
||||||
|
|
||||||
|
if ($root === null) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$pattern = $root.DIRECTORY_SEPARATOR.$prefix.'*';
|
||||||
|
$matches = glob($pattern);
|
||||||
|
|
||||||
|
if ($matches === false) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$keys = [];
|
||||||
|
|
||||||
|
foreach ($matches as $path) {
|
||||||
|
$keys[] = basename($path);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $keys;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function pathFor(string $key): string
|
||||||
|
{
|
||||||
|
return $this->rootDir.DIRECTORY_SEPARATOR.$key;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function resolvedRoot(): ?string
|
||||||
|
{
|
||||||
|
if ($this->resolvedRoot !== null) {
|
||||||
|
return $this->resolvedRoot;
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolved = @realpath($this->rootDir);
|
||||||
|
|
||||||
|
if ($resolved === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->resolvedRoot = $resolved;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function ensureRoot(): bool
|
||||||
|
{
|
||||||
|
if (is_dir($this->rootDir)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (@mkdir($this->rootDir, 0755, true)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return is_dir($this->rootDir);
|
||||||
|
}
|
||||||
|
}
|
||||||
282
src/Plugins/Tia/Fingerprint.php
Normal file
282
src/Plugins/Tia/Fingerprint.php
Normal file
@ -0,0 +1,282 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Symfony\Component\Finder\Finder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Fingerprint
|
||||||
|
{
|
||||||
|
private const int SCHEMA_VERSION = 17;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{
|
||||||
|
* structural: array<string, int|string|null>,
|
||||||
|
* environmental: array<string, int|string|null>,
|
||||||
|
* }
|
||||||
|
*/
|
||||||
|
public static function compute(string $projectRoot): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'structural' => [
|
||||||
|
'schema' => self::SCHEMA_VERSION,
|
||||||
|
'composer_lock' => self::composerLockHash($projectRoot),
|
||||||
|
'phpunit_xml' => self::trackedHash($projectRoot, 'phpunit.xml'),
|
||||||
|
'phpunit_xml_dist' => self::trackedHash($projectRoot, 'phpunit.xml.dist'),
|
||||||
|
// 'pest_factory' => self::contentHashOrNull(__DIR__.'/../../Factories/TestCaseFactory.php'),
|
||||||
|
// 'pest_method_factory' => self::contentHashOrNull(__DIR__.'/../../Factories/TestCaseMethodFactory.php'),
|
||||||
|
'vite_config' => self::viteConfigHash($projectRoot),
|
||||||
|
// 'package_json' => self::packageJsonHash($projectRoot),
|
||||||
|
'package_lock' => self::packageLockHash($projectRoot),
|
||||||
|
'js_config' => self::jsConfigHash($projectRoot),
|
||||||
|
// 'composer_json' => self::composerJsonHash($projectRoot),
|
||||||
|
],
|
||||||
|
'environmental' => [
|
||||||
|
'php_minor' => PHP_MAJOR_VERSION,
|
||||||
|
|
||||||
|
// 'extensions' => self::extensionsFingerprint($projectRoot),
|
||||||
|
// 'env_files' => self::envFilesHash($projectRoot),
|
||||||
|
],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $a
|
||||||
|
* @param array<string, mixed> $b
|
||||||
|
*/
|
||||||
|
public static function structuralMatches(array $a, array $b): bool
|
||||||
|
{
|
||||||
|
$aStructural = self::structuralOnly($a);
|
||||||
|
$bStructural = self::structuralOnly($b);
|
||||||
|
|
||||||
|
ksort($aStructural);
|
||||||
|
ksort($bStructural);
|
||||||
|
|
||||||
|
return $aStructural === $bStructural;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $stored
|
||||||
|
* @param array<string, mixed> $current
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
public static function structuralDrift(array $stored, array $current): array
|
||||||
|
{
|
||||||
|
return self::detectDrift(
|
||||||
|
self::structuralOnly($stored),
|
||||||
|
self::structuralOnly($current),
|
||||||
|
'schema',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $stored
|
||||||
|
* @param array<string, mixed> $current
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
public static function environmentalDrift(array $stored, array $current): array
|
||||||
|
{
|
||||||
|
return self::detectDrift(
|
||||||
|
self::environmentalOnly($stored),
|
||||||
|
self::environmentalOnly($current),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $a
|
||||||
|
* @param array<string, mixed> $b
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private static function detectDrift(array $a, array $b, ?string $skipKey = null): array
|
||||||
|
{
|
||||||
|
$drifts = [];
|
||||||
|
|
||||||
|
foreach ($a as $key => $value) {
|
||||||
|
if ($key === $skipKey) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (($b[$key] ?? null) !== $value) {
|
||||||
|
$drifts[] = $key;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($b as $key => $value) {
|
||||||
|
if ($key === $skipKey) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (! array_key_exists($key, $a) && $value !== null) {
|
||||||
|
$drifts[] = $key;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values(array_unique($drifts));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $fingerprint
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
private static function structuralOnly(array $fingerprint): array
|
||||||
|
{
|
||||||
|
return self::bucket($fingerprint, 'structural');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $fingerprint
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
private static function environmentalOnly(array $fingerprint): array
|
||||||
|
{
|
||||||
|
return self::bucket($fingerprint, 'environmental');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $fingerprint
|
||||||
|
* @return array<string, mixed>
|
||||||
|
*/
|
||||||
|
private static function bucket(array $fingerprint, string $key): array
|
||||||
|
{
|
||||||
|
$raw = $fingerprint[$key] ?? null;
|
||||||
|
|
||||||
|
if (! is_array($raw)) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$normalised = [];
|
||||||
|
|
||||||
|
foreach ($raw as $k => $v) {
|
||||||
|
if (is_string($k)) {
|
||||||
|
$normalised[$k] = $v;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $normalised;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function viteConfigHash(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$parts = [];
|
||||||
|
|
||||||
|
foreach (JsModuleGraph::VITE_CONFIG_NAMES as $name) {
|
||||||
|
if (! self::isTrackedByGit($projectRoot, $name)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$hash = self::contentHashOrNull($projectRoot.'/'.$name);
|
||||||
|
|
||||||
|
if ($hash !== null) {
|
||||||
|
$parts[] = $name.':'.$hash;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $parts === [] ? null : hash('xxh128', implode("\n", $parts));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function jsConfigHash(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$parts = [];
|
||||||
|
|
||||||
|
foreach (['tsconfig.json', 'tsconfig.app.json', 'jsconfig.json'] as $name) {
|
||||||
|
if (! self::isTrackedByGit($projectRoot, $name)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$hash = self::hashIfExists($projectRoot.'/'.$name);
|
||||||
|
|
||||||
|
if ($hash !== null) {
|
||||||
|
$parts[] = $name.':'.$hash;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $parts === [] ? null : hash('xxh128', implode("\n", $parts));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function composerLockHash(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
return self::trackedHash($projectRoot, 'composer.lock');
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function packageLockHash(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$parts = [];
|
||||||
|
|
||||||
|
foreach (['package-lock.json', 'pnpm-lock.yaml', 'yarn.lock', 'bun.lock', 'bun.lockb'] as $name) {
|
||||||
|
$hash = self::trackedHash($projectRoot, $name);
|
||||||
|
|
||||||
|
if ($hash !== null) {
|
||||||
|
$parts[] = $name.':'.$hash;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $parts === [] ? null : hash('xxh128', implode("\n", $parts));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function trackedHash(string $projectRoot, string $relativePath): ?string
|
||||||
|
{
|
||||||
|
if (! self::isTrackedByGit($projectRoot, $relativePath)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::hashIfExists($projectRoot.'/'.$relativePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true when the file exists and is not gitignored.
|
||||||
|
*
|
||||||
|
* Gitignored lockfiles (e.g. `package-lock.json` excluded from the repo)
|
||||||
|
* regenerate per-machine with OS-specific optional deps, which would
|
||||||
|
* otherwise force a fingerprint mismatch on every fetched baseline.
|
||||||
|
*/
|
||||||
|
private static function isTrackedByGit(string $projectRoot, string $relativePath): bool
|
||||||
|
{
|
||||||
|
if (! is_file($projectRoot.'/'.$relativePath)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static $cache = [];
|
||||||
|
|
||||||
|
$key = $projectRoot."\0".$relativePath;
|
||||||
|
|
||||||
|
if (isset($cache[$key])) {
|
||||||
|
return $cache[$key];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! is_dir($projectRoot.'/.git') && ! is_file($projectRoot.'/.git')) {
|
||||||
|
return $cache[$key] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$finder = (new Finder)
|
||||||
|
->in($projectRoot)
|
||||||
|
->depth('== 0')
|
||||||
|
->name($relativePath)
|
||||||
|
->ignoreVCSIgnored(true);
|
||||||
|
|
||||||
|
return $cache[$key] = $finder->hasResults();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function contentHashOrNull(string $path): ?string
|
||||||
|
{
|
||||||
|
if (! is_file($path)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$hash = ContentHash::of($path);
|
||||||
|
|
||||||
|
return $hash === false ? null : $hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function hashIfExists(string $path): ?string
|
||||||
|
{
|
||||||
|
if (! is_file($path)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$hash = @hash_file('xxh128', $path);
|
||||||
|
|
||||||
|
return $hash === false ? null : $hash;
|
||||||
|
}
|
||||||
|
}
|
||||||
1491
src/Plugins/Tia/Graph.php
Normal file
1491
src/Plugins/Tia/Graph.php
Normal file
File diff suppressed because it is too large
Load Diff
397
src/Plugins/Tia/JsModuleGraph.php
Normal file
397
src/Plugins/Tia/JsModuleGraph.php
Normal file
@ -0,0 +1,397 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Symfony\Component\Process\ExecutableFinder;
|
||||||
|
use Symfony\Component\Process\Process;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class JsModuleGraph
|
||||||
|
{
|
||||||
|
private const int NODE_TIMEOUT_SECONDS = 180;
|
||||||
|
|
||||||
|
private const string CACHE_FILE = 'js-module-graph.cache.json';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var list<string>
|
||||||
|
*/
|
||||||
|
public const array VITE_CONFIG_NAMES = [
|
||||||
|
'vite.config.ts',
|
||||||
|
'vite.config.js',
|
||||||
|
'vite.config.mjs',
|
||||||
|
'vite.config.cjs',
|
||||||
|
'vite.config.mts',
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Candidate page directories, in priority order. Must stay in sync with
|
||||||
|
* `PAGE_DIR_CANDIDATES` in bin/pest-tia-vite-deps.mjs.
|
||||||
|
*
|
||||||
|
* @var list<string>
|
||||||
|
*/
|
||||||
|
private const array PAGE_DIR_CANDIDATES = [
|
||||||
|
'resources/js/Pages',
|
||||||
|
'resources/js/pages',
|
||||||
|
'assets/js/Pages',
|
||||||
|
'assets/js/pages',
|
||||||
|
'assets/Pages',
|
||||||
|
'assets/pages',
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var list<string>
|
||||||
|
*/
|
||||||
|
private const array PAGE_EXTENSIONS = [
|
||||||
|
'vue', 'svelte',
|
||||||
|
'tsx', 'jsx',
|
||||||
|
'ts', 'js',
|
||||||
|
'mts', 'cts', 'mjs', 'cjs',
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, list<string>>
|
||||||
|
*/
|
||||||
|
public static function build(string $projectRoot): array
|
||||||
|
{
|
||||||
|
$result = self::resolve($projectRoot);
|
||||||
|
|
||||||
|
return $result ?? [];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, list<string>>|null
|
||||||
|
*/
|
||||||
|
public static function buildStrict(string $projectRoot): ?array
|
||||||
|
{
|
||||||
|
return self::resolve($projectRoot);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function isApplicable(string $projectRoot): bool
|
||||||
|
{
|
||||||
|
if (! self::hasViteConfig($projectRoot)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::firstExistingPagesDir($projectRoot) !== null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function firstExistingPagesDir(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
foreach (self::PAGE_DIR_CANDIDATES as $rel) {
|
||||||
|
$abs = $projectRoot.DIRECTORY_SEPARATOR.str_replace('/', DIRECTORY_SEPARATOR, $rel);
|
||||||
|
|
||||||
|
if (! is_dir($abs)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self::dirHasPageFile($abs)) {
|
||||||
|
return $abs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function dirHasPageFile(string $dir): bool
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$iterator = new \RecursiveIteratorIterator(
|
||||||
|
new \RecursiveDirectoryIterator($dir, \FilesystemIterator::SKIP_DOTS),
|
||||||
|
\RecursiveIteratorIterator::LEAVES_ONLY,
|
||||||
|
);
|
||||||
|
} catch (\UnexpectedValueException) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var \SplFileInfo $file */
|
||||||
|
foreach ($iterator as $file) {
|
||||||
|
if (! $file->isFile()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (in_array(strtolower($file->getExtension()), self::PAGE_EXTENSIONS, true)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, list<string>>|null
|
||||||
|
*/
|
||||||
|
private static function resolve(string $projectRoot): ?array
|
||||||
|
{
|
||||||
|
$fingerprint = self::fingerprint($projectRoot);
|
||||||
|
|
||||||
|
if ($fingerprint !== null) {
|
||||||
|
$cached = self::readCache($projectRoot, $fingerprint);
|
||||||
|
|
||||||
|
if ($cached !== null) {
|
||||||
|
return $cached;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$process = self::buildNodeProcess($projectRoot);
|
||||||
|
|
||||||
|
if (! $process instanceof Process) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$process->run();
|
||||||
|
|
||||||
|
if (! $process->isSuccessful()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$result = self::parseNodeOutput($process->getOutput());
|
||||||
|
|
||||||
|
if ($result !== null && $fingerprint !== null) {
|
||||||
|
self::writeCache($projectRoot, $fingerprint, $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function buildNodeProcess(string $projectRoot): ?Process
|
||||||
|
{
|
||||||
|
if (! self::hasViteConfig($projectRoot)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! is_dir($projectRoot.DIRECTORY_SEPARATOR.'node_modules'.DIRECTORY_SEPARATOR.'vite')) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$nodeBinary = (new ExecutableFinder)->find('node');
|
||||||
|
|
||||||
|
if ($nodeBinary === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$helperPath = dirname(__DIR__, 3).DIRECTORY_SEPARATOR.'bin'.DIRECTORY_SEPARATOR.'pest-tia-vite-deps.mjs';
|
||||||
|
|
||||||
|
if (! is_file($helperPath)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$process = new Process([$nodeBinary, $helperPath, $projectRoot], $projectRoot);
|
||||||
|
$process->setTimeout(self::NODE_TIMEOUT_SECONDS);
|
||||||
|
|
||||||
|
return $process;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, list<string>>|null
|
||||||
|
*/
|
||||||
|
private static function parseNodeOutput(string $output): ?array
|
||||||
|
{
|
||||||
|
/** @var mixed $decoded */
|
||||||
|
$decoded = json_decode($output, true);
|
||||||
|
|
||||||
|
if (! is_array($decoded)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($decoded as $path => $components) {
|
||||||
|
if (! is_string($path)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (! is_array($components)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$names = [];
|
||||||
|
|
||||||
|
foreach ($components as $component) {
|
||||||
|
if (is_string($component) && $component !== '') {
|
||||||
|
$names[] = $component;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($names !== []) {
|
||||||
|
sort($names);
|
||||||
|
$out[$path] = $names;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ksort($out);
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function fingerprint(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$parts = [];
|
||||||
|
|
||||||
|
foreach (self::VITE_CONFIG_NAMES as $name) {
|
||||||
|
$path = $projectRoot.DIRECTORY_SEPARATOR.$name;
|
||||||
|
|
||||||
|
if (! is_file($path)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$stat = @stat($path);
|
||||||
|
$bytes = @file_get_contents($path);
|
||||||
|
|
||||||
|
$parts[] = 'config:'.$name
|
||||||
|
.':'.($stat === false ? '0' : (string) $stat['mtime'])
|
||||||
|
.':'.($stat === false ? '0' : (string) $stat['size'])
|
||||||
|
.':'.($bytes === false ? '' : hash('sha256', $bytes));
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($parts === []) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$override = getenv('TIA_VITE_PAGES_DIR');
|
||||||
|
|
||||||
|
if (is_string($override) && $override !== '') {
|
||||||
|
$parts[] = 'pagesDirOverride:'.$override;
|
||||||
|
}
|
||||||
|
|
||||||
|
$pagesDir = self::firstExistingPagesDir($projectRoot);
|
||||||
|
|
||||||
|
if ($pagesDir !== null) {
|
||||||
|
$parts[] = 'pagesDir:'.str_replace($projectRoot.DIRECTORY_SEPARATOR, '', $pagesDir);
|
||||||
|
}
|
||||||
|
|
||||||
|
$jsRoot = $pagesDir !== null ? dirname($pagesDir) : null;
|
||||||
|
|
||||||
|
if ($jsRoot !== null && is_dir($jsRoot)) {
|
||||||
|
$entries = [];
|
||||||
|
|
||||||
|
$iterator = new \RecursiveIteratorIterator(
|
||||||
|
new \RecursiveDirectoryIterator($jsRoot, \FilesystemIterator::SKIP_DOTS),
|
||||||
|
\RecursiveIteratorIterator::LEAVES_ONLY,
|
||||||
|
);
|
||||||
|
|
||||||
|
/** @var \SplFileInfo $file */
|
||||||
|
foreach ($iterator as $file) {
|
||||||
|
if (! $file->isFile()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$entries[] = $file->getPathname()
|
||||||
|
.':'.$file->getSize()
|
||||||
|
.':'.$file->getMTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
sort($entries);
|
||||||
|
|
||||||
|
$parts[] = 'js:'.hash('sha256', implode("\n", $entries));
|
||||||
|
}
|
||||||
|
|
||||||
|
return hash('sha256', implode('|', $parts));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, list<string>>|null
|
||||||
|
*/
|
||||||
|
private static function readCache(string $projectRoot, string $fingerprint): ?array
|
||||||
|
{
|
||||||
|
$path = self::cachePath($projectRoot);
|
||||||
|
|
||||||
|
if (! is_file($path)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$raw = @file_get_contents($path);
|
||||||
|
|
||||||
|
if ($raw === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var mixed $decoded */
|
||||||
|
$decoded = json_decode($raw, true);
|
||||||
|
|
||||||
|
if (! is_array($decoded)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (($decoded['fingerprint'] ?? null) !== $fingerprint) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$graph = $decoded['graph'] ?? null;
|
||||||
|
|
||||||
|
if (! is_array($graph)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($graph as $key => $value) {
|
||||||
|
if (! is_string($key)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (! is_array($value)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$names = [];
|
||||||
|
|
||||||
|
foreach ($value as $name) {
|
||||||
|
if (is_string($name) && $name !== '') {
|
||||||
|
$names[] = $name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$out[$key] = $names;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, list<string>> $graph
|
||||||
|
*/
|
||||||
|
private static function writeCache(string $projectRoot, string $fingerprint, array $graph): void
|
||||||
|
{
|
||||||
|
$path = self::cachePath($projectRoot);
|
||||||
|
$dir = dirname($path);
|
||||||
|
|
||||||
|
if (! is_dir($dir) && ! @mkdir($dir, 0755, true) && ! is_dir($dir)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$payload = json_encode([
|
||||||
|
'fingerprint' => $fingerprint,
|
||||||
|
'graph' => $graph,
|
||||||
|
]);
|
||||||
|
|
||||||
|
if ($payload === false) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tmp = $path.'.tmp.'.bin2hex(random_bytes(4));
|
||||||
|
|
||||||
|
if (@file_put_contents($tmp, $payload) === false) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! @rename($tmp, $path)) {
|
||||||
|
@unlink($tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function cachePath(string $projectRoot): string
|
||||||
|
{
|
||||||
|
return Storage::tempDir($projectRoot).DIRECTORY_SEPARATOR.self::CACHE_FILE;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function hasViteConfig(string $projectRoot): bool
|
||||||
|
{
|
||||||
|
foreach (self::VITE_CONFIG_NAMES as $name) {
|
||||||
|
if (is_file($projectRoot.DIRECTORY_SEPARATOR.$name)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
355
src/Plugins/Tia/Recorder.php
Normal file
355
src/Plugins/Tia/Recorder.php
Normal file
@ -0,0 +1,355 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\TestSuite;
|
||||||
|
use ReflectionClass;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class Recorder
|
||||||
|
{
|
||||||
|
private ?string $currentTestFile = null;
|
||||||
|
|
||||||
|
/** @var array<string, array<string, true>> */
|
||||||
|
private array $perTestFiles = [];
|
||||||
|
|
||||||
|
/** @var array<string, array<string, true>> */
|
||||||
|
private array $perTestTables = [];
|
||||||
|
|
||||||
|
/** @var array<string, array<string, true>> */
|
||||||
|
private array $perTestInertiaComponents = [];
|
||||||
|
|
||||||
|
/** @var array<string, true> */
|
||||||
|
private array $perTestUsesDatabase = [];
|
||||||
|
|
||||||
|
/** @var array<string, string|null> */
|
||||||
|
private array $classFileCache = [];
|
||||||
|
|
||||||
|
/** @var array<string, bool> */
|
||||||
|
private array $classUsesDatabaseCache = [];
|
||||||
|
|
||||||
|
private bool $active = false;
|
||||||
|
|
||||||
|
private bool $driverChecked = false;
|
||||||
|
|
||||||
|
private bool $driverAvailable = false;
|
||||||
|
|
||||||
|
private string $driver = 'none';
|
||||||
|
|
||||||
|
private ?SourceScope $sourceScope = null;
|
||||||
|
|
||||||
|
public function activate(): void
|
||||||
|
{
|
||||||
|
$this->active = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isActive(): bool
|
||||||
|
{
|
||||||
|
return $this->active;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function driverAvailable(): bool
|
||||||
|
{
|
||||||
|
if (! $this->driverChecked) {
|
||||||
|
if (function_exists('pcov\\start')) {
|
||||||
|
$this->driver = 'pcov';
|
||||||
|
$this->driverAvailable = true;
|
||||||
|
} elseif (function_exists('xdebug_start_code_coverage') && function_exists('xdebug_info')) {
|
||||||
|
$modes = \xdebug_info('mode');
|
||||||
|
|
||||||
|
if (is_array($modes) && in_array('coverage', $modes, true)) {
|
||||||
|
$this->driver = 'xdebug';
|
||||||
|
$this->driverAvailable = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->driverChecked = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->driverAvailable;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function beginTest(string $className, string $methodName, string $fallbackFile): void
|
||||||
|
{
|
||||||
|
if (! $this->active || ! $this->driverAvailable()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->currentTestFile !== null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$file = $this->resolveTestFile($className, $fallbackFile);
|
||||||
|
|
||||||
|
if ($file === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->currentTestFile = $file;
|
||||||
|
|
||||||
|
if ($this->classUsesDatabase($className)) {
|
||||||
|
$this->perTestUsesDatabase[$file] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->driver === 'pcov') {
|
||||||
|
\pcov\clear();
|
||||||
|
\pcov\start();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
\xdebug_start_code_coverage();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function endTest(): void
|
||||||
|
{
|
||||||
|
if (! $this->active || ! $this->driverAvailable() || $this->currentTestFile === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->driver === 'pcov') {
|
||||||
|
\pcov\stop();
|
||||||
|
|
||||||
|
$scope = $this->sourceScope();
|
||||||
|
$filesToCollectCoverageFor = [];
|
||||||
|
|
||||||
|
foreach (\pcov\waiting() as $file) {
|
||||||
|
if (is_string($file) && $scope->contains($file)) {
|
||||||
|
$filesToCollectCoverageFor[] = $file;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var array<string, mixed> $data */
|
||||||
|
$data = \pcov\collect(\pcov\inclusive, $filesToCollectCoverageFor);
|
||||||
|
|
||||||
|
$coveredFiles = $this->filesWithExecutedLines($data);
|
||||||
|
} else {
|
||||||
|
/** @var array<string, mixed> $data */
|
||||||
|
$data = \xdebug_get_code_coverage();
|
||||||
|
\xdebug_stop_code_coverage(true);
|
||||||
|
|
||||||
|
$coveredFiles = array_keys($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($coveredFiles as $sourceFile) {
|
||||||
|
$this->perTestFiles[$this->currentTestFile][$sourceFile] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->currentTestFile = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function linkSource(string $sourceFile): void
|
||||||
|
{
|
||||||
|
if (! $this->active) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->currentTestFile === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($sourceFile === '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->perTestFiles[$this->currentTestFile][$sourceFile] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function classUsesDatabase(string $className): bool
|
||||||
|
{
|
||||||
|
if (array_key_exists($className, $this->classUsesDatabaseCache)) {
|
||||||
|
return $this->classUsesDatabaseCache[$className];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! class_exists($className, false)) {
|
||||||
|
return $this->classUsesDatabaseCache[$className] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static $needles = [
|
||||||
|
'Illuminate\\Foundation\\Testing\\RefreshDatabase' => true,
|
||||||
|
'Illuminate\\Foundation\\Testing\\DatabaseMigrations' => true,
|
||||||
|
'Illuminate\\Foundation\\Testing\\DatabaseTransactions' => true,
|
||||||
|
];
|
||||||
|
|
||||||
|
$reflection = new ReflectionClass($className);
|
||||||
|
|
||||||
|
do {
|
||||||
|
foreach (array_keys($reflection->getTraits()) as $traitName) {
|
||||||
|
if (isset($needles[$traitName])) {
|
||||||
|
return $this->classUsesDatabaseCache[$className] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$reflection = $reflection->getParentClass();
|
||||||
|
} while ($reflection !== false && ! $reflection->isInternal());
|
||||||
|
|
||||||
|
return $this->classUsesDatabaseCache[$className] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function linkTable(string $table): void
|
||||||
|
{
|
||||||
|
if (! $this->active) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->currentTestFile === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($table === '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->perTestTables[$this->currentTestFile][strtolower($table)] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function linkInertiaComponent(string $component): void
|
||||||
|
{
|
||||||
|
if (! $this->active) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->currentTestFile === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($component === '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->perTestInertiaComponents[$this->currentTestFile][$component] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array<string, array<int, string>> */
|
||||||
|
public function perTestFiles(): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($this->perTestFiles as $testFile => $sources) {
|
||||||
|
$out[$testFile] = array_keys($sources);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array<string, array<int, string>> */
|
||||||
|
public function perTestTables(): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($this->perTestTables as $testFile => $tables) {
|
||||||
|
$names = array_keys($tables);
|
||||||
|
sort($names);
|
||||||
|
$out[$testFile] = $names;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array<string, array<int, string>> */
|
||||||
|
public function perTestInertiaComponents(): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($this->perTestInertiaComponents as $testFile => $components) {
|
||||||
|
$names = array_keys($components);
|
||||||
|
sort($names);
|
||||||
|
$out[$testFile] = $names;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array<string, true> */
|
||||||
|
public function perTestUsesDatabase(): array
|
||||||
|
{
|
||||||
|
return $this->perTestUsesDatabase;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function resolveTestFile(string $className, string $fallbackFile): ?string
|
||||||
|
{
|
||||||
|
if (array_key_exists($className, $this->classFileCache)) {
|
||||||
|
$file = $this->classFileCache[$className];
|
||||||
|
} else {
|
||||||
|
$file = $this->readPestFilename($className);
|
||||||
|
$this->classFileCache[$className] = $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($file !== null) {
|
||||||
|
return $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($fallbackFile !== '' && $fallbackFile !== 'unknown' && ! str_contains($fallbackFile, "eval()'d")) {
|
||||||
|
return $fallbackFile;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function readPestFilename(string $className): ?string
|
||||||
|
{
|
||||||
|
if (! class_exists($className, false)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(property_exists($className, '__filename') && is_string($className::$__filename));
|
||||||
|
|
||||||
|
return $className::$__filename;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $data
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private function filesWithExecutedLines(array $data): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($data as $file => $lines) {
|
||||||
|
if (! is_array($lines)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$covered = [];
|
||||||
|
foreach ($lines as $line => $count) {
|
||||||
|
if (is_int($count) && $count > 0) {
|
||||||
|
$covered[] = $line;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($covered === []) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$lineKeys = array_keys($lines);
|
||||||
|
if ($lineKeys !== [] && count($covered) === 1 && $covered[0] === max($lineKeys)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$out[] = $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function sourceScope(): SourceScope
|
||||||
|
{
|
||||||
|
return $this->sourceScope ??= SourceScope::fromProjectRoot(TestSuite::getInstance()->rootPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reset(): void
|
||||||
|
{
|
||||||
|
$this->currentTestFile = null;
|
||||||
|
$this->perTestFiles = [];
|
||||||
|
$this->perTestTables = [];
|
||||||
|
$this->perTestInertiaComponents = [];
|
||||||
|
$this->perTestUsesDatabase = [];
|
||||||
|
$this->classFileCache = [];
|
||||||
|
$this->classUsesDatabaseCache = [];
|
||||||
|
$this->sourceScope = null;
|
||||||
|
$this->active = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
149
src/Plugins/Tia/ResultCollector.php
Normal file
149
src/Plugins/Tia/ResultCollector.php
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use PHPUnit\Framework\TestStatus\TestStatus;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class ResultCollector
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var array<string, array{status: int, message: string, time: float, assertions: int, file?: string}>
|
||||||
|
*/
|
||||||
|
private array $results = [];
|
||||||
|
|
||||||
|
private ?string $currentTestId = null;
|
||||||
|
|
||||||
|
private ?string $currentTestFile = null;
|
||||||
|
|
||||||
|
private ?float $startTime = null;
|
||||||
|
|
||||||
|
public function testPrepared(string $testId, ?string $testFile = null): void
|
||||||
|
{
|
||||||
|
$this->currentTestId = $testId;
|
||||||
|
$this->currentTestFile = $testFile;
|
||||||
|
$this->startTime = microtime(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testPassed(): void
|
||||||
|
{
|
||||||
|
if ($this->currentTestId === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->record(TestStatus::success());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testFailed(string $message): void
|
||||||
|
{
|
||||||
|
if ($this->currentTestId === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->record(TestStatus::failure($message));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testErrored(string $message): void
|
||||||
|
{
|
||||||
|
if ($this->currentTestId === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->record(TestStatus::error($message));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testSkipped(string $message): void
|
||||||
|
{
|
||||||
|
if ($this->currentTestId === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->record(TestStatus::skipped($message));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testIncomplete(string $message): void
|
||||||
|
{
|
||||||
|
if ($this->currentTestId === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->record(TestStatus::incomplete($message));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testRisky(string $message): void
|
||||||
|
{
|
||||||
|
if ($this->currentTestId === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->record(TestStatus::risky($message));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, array{status: int, message: string, time: float, assertions: int, file?: string}>
|
||||||
|
*/
|
||||||
|
public function all(): array
|
||||||
|
{
|
||||||
|
return $this->results;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function recordAssertions(string $testId, int $assertions): void
|
||||||
|
{
|
||||||
|
if (isset($this->results[$testId])) {
|
||||||
|
$this->results[$testId]['assertions'] = $assertions;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, array{status: int, message: string, time: float, assertions: int, file?: string}> $results
|
||||||
|
*/
|
||||||
|
public function merge(array $results): void
|
||||||
|
{
|
||||||
|
foreach ($results as $testId => $result) {
|
||||||
|
$this->results[$testId] = $result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reset(): void
|
||||||
|
{
|
||||||
|
$this->results = [];
|
||||||
|
$this->currentTestId = null;
|
||||||
|
$this->currentTestFile = null;
|
||||||
|
$this->startTime = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function finishTest(): void
|
||||||
|
{
|
||||||
|
$this->currentTestId = null;
|
||||||
|
$this->currentTestFile = null;
|
||||||
|
$this->startTime = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function record(TestStatus $status): void
|
||||||
|
{
|
||||||
|
if ($this->currentTestId === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$time = $this->startTime !== null
|
||||||
|
? round(microtime(true) - $this->startTime, 3)
|
||||||
|
: 0.0;
|
||||||
|
|
||||||
|
$existing = $this->results[$this->currentTestId] ?? null;
|
||||||
|
|
||||||
|
$this->results[$this->currentTestId] = [
|
||||||
|
'status' => $status->asInt(),
|
||||||
|
'message' => $status->message(),
|
||||||
|
'time' => $time,
|
||||||
|
'assertions' => $existing['assertions'] ?? 0,
|
||||||
|
];
|
||||||
|
|
||||||
|
if ($this->currentTestFile !== null) {
|
||||||
|
$this->results[$this->currentTestId]['file'] = $this->currentTestFile;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
196
src/Plugins/Tia/SourceScope.php
Normal file
196
src/Plugins/Tia/SourceScope.php
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use PHPUnit\TextUI\Configuration\Registry;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class SourceScope
|
||||||
|
{
|
||||||
|
/** @var array<string, bool> */
|
||||||
|
private array $containsCache = [];
|
||||||
|
|
||||||
|
private const array TOP_LEVEL_NOISE = [
|
||||||
|
'vendor',
|
||||||
|
'node_modules',
|
||||||
|
'.git',
|
||||||
|
'.idea',
|
||||||
|
'.vscode',
|
||||||
|
'.github',
|
||||||
|
'.pest',
|
||||||
|
'.phpunit.cache',
|
||||||
|
'.cache',
|
||||||
|
];
|
||||||
|
|
||||||
|
private const array NESTED_NOISE = [
|
||||||
|
'storage/framework',
|
||||||
|
'storage/logs',
|
||||||
|
'bootstrap/cache',
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param list<string> $includes Absolute, normalised directory paths.
|
||||||
|
* @param list<string> $excludes Absolute, normalised directory paths.
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
private readonly array $includes,
|
||||||
|
private readonly array $excludes,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public static function fromProjectRoot(string $projectRoot): self
|
||||||
|
{
|
||||||
|
$phpunitIncludes = [];
|
||||||
|
$phpunitExcludes = [];
|
||||||
|
|
||||||
|
try {
|
||||||
|
$source = Registry::get()->source();
|
||||||
|
|
||||||
|
foreach ($source->includeDirectories() as $dir) {
|
||||||
|
$phpunitIncludes[] = self::normalise($dir->path());
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($source->excludeDirectories() as $dir) {
|
||||||
|
$phpunitExcludes[] = self::normalise($dir->path());
|
||||||
|
}
|
||||||
|
} catch (Throwable) {
|
||||||
|
// Registry not initialized — fall back to project-root scanning.
|
||||||
|
}
|
||||||
|
|
||||||
|
$rootIncludes = self::topLevelProjectDirs($projectRoot);
|
||||||
|
|
||||||
|
$includes = array_values(array_unique([...$phpunitIncludes, ...$rootIncludes]));
|
||||||
|
$excludes = array_values(array_unique([
|
||||||
|
...$phpunitExcludes,
|
||||||
|
...self::nestedNoiseDirs($projectRoot),
|
||||||
|
]));
|
||||||
|
|
||||||
|
if ($includes === []) {
|
||||||
|
$includes = [self::normalise($projectRoot)];
|
||||||
|
}
|
||||||
|
|
||||||
|
return new self($includes, $excludes);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string> Absolute, normalised paths to testsuite directories and files declared in phpunit.xml.
|
||||||
|
*/
|
||||||
|
public static function testPaths(): array
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$suites = Registry::get()->testSuite();
|
||||||
|
} catch (Throwable) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
$out = [];
|
||||||
|
foreach ($suites as $suite) {
|
||||||
|
foreach ($suite->directories() as $directory) {
|
||||||
|
$out[] = self::normalise($directory->path());
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($suite->files() as $file) {
|
||||||
|
$out[] = self::normalise($file->path());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values(array_unique($out));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function contains(string $absoluteFile): bool
|
||||||
|
{
|
||||||
|
if (isset($this->containsCache[$absoluteFile])) {
|
||||||
|
return $this->containsCache[$absoluteFile];
|
||||||
|
}
|
||||||
|
|
||||||
|
$real = @realpath($absoluteFile);
|
||||||
|
$candidate = $real === false ? $absoluteFile : $real;
|
||||||
|
$candidate = self::normalise($candidate);
|
||||||
|
|
||||||
|
foreach ($this->excludes as $excluded) {
|
||||||
|
if ($this->startsWithDir($candidate, $excluded)) {
|
||||||
|
return $this->containsCache[$absoluteFile] = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($this->includes as $included) {
|
||||||
|
if ($this->startsWithDir($candidate, $included)) {
|
||||||
|
return $this->containsCache[$absoluteFile] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->containsCache[$absoluteFile] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private static function topLevelProjectDirs(string $projectRoot): array
|
||||||
|
{
|
||||||
|
$entries = @scandir($projectRoot);
|
||||||
|
|
||||||
|
if ($entries === false) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach ($entries as $entry) {
|
||||||
|
if ($entry === '.') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($entry === '..') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (in_array($entry, self::TOP_LEVEL_NOISE, true)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($entry !== '' && $entry[0] === '.') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$abs = $projectRoot.DIRECTORY_SEPARATOR.$entry;
|
||||||
|
|
||||||
|
if (! is_dir($abs)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$out[] = self::normalise(@realpath($abs) ?: $abs);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string>
|
||||||
|
*/
|
||||||
|
private static function nestedNoiseDirs(string $projectRoot): array
|
||||||
|
{
|
||||||
|
$out = [];
|
||||||
|
|
||||||
|
foreach (self::NESTED_NOISE as $relative) {
|
||||||
|
$abs = $projectRoot.DIRECTORY_SEPARATOR.str_replace('/', DIRECTORY_SEPARATOR, $relative);
|
||||||
|
$out[] = self::normalise(@realpath($abs) ?: $abs);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function normalise(string $path): string
|
||||||
|
{
|
||||||
|
return rtrim($path, '/\\');
|
||||||
|
}
|
||||||
|
|
||||||
|
private function startsWithDir(string $candidate, string $dir): bool
|
||||||
|
{
|
||||||
|
if ($candidate === $dir) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return str_starts_with($candidate, $dir.DIRECTORY_SEPARATOR);
|
||||||
|
}
|
||||||
|
}
|
||||||
146
src/Plugins/Tia/Storage.php
Normal file
146
src/Plugins/Tia/Storage.php
Normal file
@ -0,0 +1,146 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class Storage
|
||||||
|
{
|
||||||
|
public static function tempDir(string $projectRoot): string
|
||||||
|
{
|
||||||
|
$home = self::homeDir();
|
||||||
|
|
||||||
|
if ($home === null) {
|
||||||
|
return $projectRoot
|
||||||
|
.DIRECTORY_SEPARATOR.'.pest'
|
||||||
|
.DIRECTORY_SEPARATOR.'tia';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $home
|
||||||
|
.DIRECTORY_SEPARATOR.'.pest'
|
||||||
|
.DIRECTORY_SEPARATOR.'tia'
|
||||||
|
.DIRECTORY_SEPARATOR.self::projectKey($projectRoot);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function purge(string $projectRoot): void
|
||||||
|
{
|
||||||
|
$dir = self::tempDir($projectRoot);
|
||||||
|
|
||||||
|
if (! is_dir($dir)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
self::removeRecursive($dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function removeRecursive(string $dir): void
|
||||||
|
{
|
||||||
|
$entries = @scandir($dir);
|
||||||
|
|
||||||
|
if ($entries === false) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($entries as $entry) {
|
||||||
|
if ($entry === '.') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($entry === '..') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$path = $dir.DIRECTORY_SEPARATOR.$entry;
|
||||||
|
|
||||||
|
if (is_dir($path) && ! is_link($path)) {
|
||||||
|
self::removeRecursive($path);
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
@unlink($path);
|
||||||
|
}
|
||||||
|
|
||||||
|
@rmdir($dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function homeDir(): ?string
|
||||||
|
{
|
||||||
|
foreach (['HOME', 'USERPROFILE'] as $key) {
|
||||||
|
$value = getenv($key);
|
||||||
|
|
||||||
|
if (is_string($value) && $value !== '' && is_dir($value)) {
|
||||||
|
return rtrim($value, '/\\');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* `git@github.com:foo/bar.git`, `ssh://git@github.com/foo/bar`
|
||||||
|
*/
|
||||||
|
private static function projectKey(string $projectRoot): string
|
||||||
|
{
|
||||||
|
$origin = self::originIdentity($projectRoot);
|
||||||
|
|
||||||
|
$realpath = @realpath($projectRoot);
|
||||||
|
$input = $origin ?? ($realpath === false ? $projectRoot : $realpath);
|
||||||
|
|
||||||
|
$hash = substr(hash('sha256', $input), 0, 16);
|
||||||
|
$slug = self::slug(basename($projectRoot));
|
||||||
|
|
||||||
|
return $slug === '' ? $hash : $slug.'-'.$hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function originIdentity(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$url = self::rawOriginUrl($projectRoot);
|
||||||
|
|
||||||
|
if ($url === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// git@host:org/repo(.git)
|
||||||
|
if (preg_match('#^[\w.-]+@([\w.-]+):([\w./-]+?)(?:\.git)?/?$#', $url, $m) === 1) {
|
||||||
|
return strtolower($m[1].'/'.$m[2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// scheme://[user@]host[:port]/org/repo(.git) — https, ssh, git, file
|
||||||
|
if (preg_match('#^[a-z]+://(?:[^@/]+@)?([^/:]+)(?::\d+)?/([\w./-]+?)(?:\.git)?/?$#i', $url, $m) === 1) {
|
||||||
|
return strtolower($m[1].'/'.$m[2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return strtolower($url);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function rawOriginUrl(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$config = $projectRoot.DIRECTORY_SEPARATOR.'.git'.DIRECTORY_SEPARATOR.'config';
|
||||||
|
|
||||||
|
if (! is_file($config)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$raw = @file_get_contents($config);
|
||||||
|
|
||||||
|
if ($raw === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (preg_match('/\[remote "origin"\][^\[]*?url\s*=\s*(\S+)/s', $raw, $match) === 1) {
|
||||||
|
return trim($match[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function slug(string $name): string
|
||||||
|
{
|
||||||
|
$slug = strtolower($name);
|
||||||
|
$slug = preg_replace('/[^a-z0-9]+/', '-', $slug) ?? '';
|
||||||
|
|
||||||
|
return trim($slug, '-');
|
||||||
|
}
|
||||||
|
}
|
||||||
136
src/Plugins/Tia/TableExtractor.php
Normal file
136
src/Plugins/Tia/TableExtractor.php
Normal file
@ -0,0 +1,136 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class TableExtractor
|
||||||
|
{
|
||||||
|
private const array DML_PREFIXES = ['select', 'insert', 'update', 'delete'];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string> Sorted, deduped table names referenced by the
|
||||||
|
*/
|
||||||
|
public static function fromSql(string $sql): array
|
||||||
|
{
|
||||||
|
$trimmed = ltrim($sql);
|
||||||
|
|
||||||
|
if ($trimmed === '') {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$prefix = strtolower(substr($trimmed, 0, 6));
|
||||||
|
|
||||||
|
$matched = false;
|
||||||
|
foreach (self::DML_PREFIXES as $dml) {
|
||||||
|
if (str_starts_with($prefix, $dml)) {
|
||||||
|
$matched = true;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $matched) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$pattern = '/(?:\bfrom|\binto|\bupdate|\bjoin)\s+(?:"([^"]+)"|`([^`]+)`|\[([^\]]+)\]|(\w+))/i';
|
||||||
|
|
||||||
|
if (preg_match_all($pattern, $sql, $matches) === false) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$tables = [];
|
||||||
|
|
||||||
|
for ($i = 0, $n = count($matches[0]); $i < $n; $i++) {
|
||||||
|
$name = $matches[1][$i] !== ''
|
||||||
|
? $matches[1][$i]
|
||||||
|
: ($matches[2][$i] !== ''
|
||||||
|
? $matches[2][$i]
|
||||||
|
: ($matches[3][$i] !== ''
|
||||||
|
? $matches[3][$i]
|
||||||
|
: $matches[4][$i]));
|
||||||
|
if ($name === '') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (self::isSchemaMeta($name)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tables[strtolower($name)] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$out = array_keys($tables);
|
||||||
|
sort($out);
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return list<string> Table names referenced by `Schema::` calls,
|
||||||
|
*/
|
||||||
|
public static function fromMigrationSource(string $php): array
|
||||||
|
{
|
||||||
|
$tables = [];
|
||||||
|
|
||||||
|
$schemaPattern = '/Schema::\s*(?:create|table|drop|dropIfExists|dropColumn|dropColumns|rename)\s*\(\s*[\'"]([^\'"]+)[\'"](?:\s*,\s*[\'"]([^\'"]+)[\'"])?/';
|
||||||
|
|
||||||
|
if (preg_match_all($schemaPattern, $php, $matches) !== false) {
|
||||||
|
foreach ($matches[1] as $i => $primary) {
|
||||||
|
$tables[strtolower($primary)] = true;
|
||||||
|
|
||||||
|
$secondary = $matches[2][$i] ?? '';
|
||||||
|
if ($secondary !== '') {
|
||||||
|
$tables[strtolower($secondary)] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$ddlPattern = '/(?:CREATE|ALTER|DROP|TRUNCATE|RENAME)\s+TABLE(?:\s+IF\s+(?:NOT\s+)?EXISTS)?\s+["`\[]?(\w+)["`\]]?/i';
|
||||||
|
|
||||||
|
if (preg_match_all($ddlPattern, $php, $matches) !== false) {
|
||||||
|
foreach ($matches[1] as $primary) {
|
||||||
|
$lower = strtolower($primary);
|
||||||
|
if (! self::isSchemaMeta($lower)) {
|
||||||
|
$tables[$lower] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$dmlPatterns = [
|
||||||
|
'/INSERT\s+(?:IGNORE\s+)?INTO\s+["`\[]?(\w+)["`\]]?/i',
|
||||||
|
'/UPDATE\s+["`\[]?(\w+)["`\]]?\s+SET\b/i',
|
||||||
|
'/DELETE\s+FROM\s+["`\[]?(\w+)["`\]]?/i',
|
||||||
|
'/DB::table\(\s*[\'"]([^\'"]+)[\'"]\s*\)/',
|
||||||
|
];
|
||||||
|
|
||||||
|
foreach ($dmlPatterns as $pattern) {
|
||||||
|
if (preg_match_all($pattern, $php, $matches) === false) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
foreach ($matches[1] as $name) {
|
||||||
|
$lower = strtolower($name);
|
||||||
|
if (! self::isSchemaMeta($lower)) {
|
||||||
|
$tables[$lower] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$out = array_keys($tables);
|
||||||
|
sort($out);
|
||||||
|
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function isSchemaMeta(string $name): bool
|
||||||
|
{
|
||||||
|
$lower = strtolower($name);
|
||||||
|
|
||||||
|
return in_array($lower, ['sqlite_master', 'sqlite_sequence', 'migrations'], true)
|
||||||
|
|| str_starts_with($lower, 'pg_')
|
||||||
|
|| str_starts_with($lower, 'information_schema');
|
||||||
|
}
|
||||||
|
}
|
||||||
86
src/Plugins/Tia/TableTracker.php
Normal file
86
src/Plugins/Tia/TableTracker.php
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class TableTracker
|
||||||
|
{
|
||||||
|
private const string CONTAINER_CLASS = '\\Illuminate\\Container\\Container';
|
||||||
|
|
||||||
|
private const string MARKER = 'pest.tia.table-tracker-armed';
|
||||||
|
|
||||||
|
public static function arm(Recorder $recorder): void
|
||||||
|
{
|
||||||
|
if (! $recorder->isActive()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$containerClass = self::CONTAINER_CLASS;
|
||||||
|
|
||||||
|
if (! class_exists($containerClass)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var object $app */
|
||||||
|
$app = $containerClass::getInstance();
|
||||||
|
|
||||||
|
if (! method_exists($app, 'bound') || ! method_exists($app, 'make') || ! method_exists($app, 'instance')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($app->bound(self::MARKER)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $app->bound('db')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$app->instance(self::MARKER, true);
|
||||||
|
|
||||||
|
$listener = static function (object $query) use ($recorder): void {
|
||||||
|
if (! property_exists($query, 'sql')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var mixed $sql */
|
||||||
|
$sql = $query->sql;
|
||||||
|
|
||||||
|
if (! is_string($sql) || $sql === '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (TableExtractor::fromSql($sql) as $table) {
|
||||||
|
$recorder->linkTable($table);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/** @var object $db */
|
||||||
|
$db = $app->make('db');
|
||||||
|
|
||||||
|
if (is_callable([$db, 'listen'])) {
|
||||||
|
/** @var callable $listen */
|
||||||
|
$listen = [$db, 'listen'];
|
||||||
|
$listen($listener);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $app->bound('events')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @var object $events */
|
||||||
|
$events = $app->make('events');
|
||||||
|
|
||||||
|
if (! method_exists($events, 'listen')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$events->listen('Illuminate\\Database\\Events\\QueryExecuted', $listener);
|
||||||
|
}
|
||||||
|
}
|
||||||
163
src/Plugins/Tia/TestPaths.php
Normal file
163
src/Plugins/Tia/TestPaths.php
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\TestSuite;
|
||||||
|
use PHPUnit\TextUI\Configuration\Registry;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolves the set of project-relative paths that are considered test files,
|
||||||
|
* driven by phpunit.xml's <testsuites>. Falls back to the runtime TestSuite
|
||||||
|
* configuration when no config file is present.
|
||||||
|
*
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class TestPaths
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param list<string> $directories Project-relative directory prefixes (no trailing slash).
|
||||||
|
* @param list<string> $files Project-relative file paths.
|
||||||
|
* @param list<string> $suffixes Filename suffixes (e.g. '.php').
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
private array $directories,
|
||||||
|
private array $files,
|
||||||
|
private array $suffixes,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public static function fromProjectRoot(string $projectRoot): self
|
||||||
|
{
|
||||||
|
$directories = [];
|
||||||
|
$files = [];
|
||||||
|
$suffixes = [];
|
||||||
|
|
||||||
|
try {
|
||||||
|
$configuration = Registry::get();
|
||||||
|
|
||||||
|
foreach ($configuration->testSuite() as $suite) {
|
||||||
|
foreach ($suite->directories() as $directory) {
|
||||||
|
$rel = self::toRelative($directory->path(), $projectRoot);
|
||||||
|
|
||||||
|
if ($rel !== null) {
|
||||||
|
$directories[] = $rel;
|
||||||
|
}
|
||||||
|
|
||||||
|
$suffix = $directory->suffix();
|
||||||
|
|
||||||
|
if ($suffix !== '') {
|
||||||
|
$suffixes[] = str_starts_with($suffix, '.') ? $suffix : '.'.$suffix;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($suite->files() as $file) {
|
||||||
|
$rel = self::toRelative($file->path(), $projectRoot);
|
||||||
|
|
||||||
|
if ($rel !== null) {
|
||||||
|
$files[] = $rel;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($suffixes === []) {
|
||||||
|
foreach ($configuration->testSuffixes() as $suffix) {
|
||||||
|
$suffixes[] = str_starts_with($suffix, '.') ? $suffix : '.'.$suffix;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Throwable) {
|
||||||
|
// Registry not initialized — fall through to defaults.
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($suffixes === []) {
|
||||||
|
$suffixes = ['.php'];
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($directories === [] && $files === []) {
|
||||||
|
$fallback = self::testSuiteFallback($projectRoot);
|
||||||
|
|
||||||
|
if ($fallback !== null) {
|
||||||
|
$directories[] = $fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return new self(
|
||||||
|
array_values(array_unique($directories)),
|
||||||
|
array_values(array_unique($files)),
|
||||||
|
array_values(array_unique($suffixes)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isTestFile(string $relativePath): bool
|
||||||
|
{
|
||||||
|
if (in_array($relativePath, $this->files, true)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$matchesSuffix = false;
|
||||||
|
foreach ($this->suffixes as $suffix) {
|
||||||
|
if (str_ends_with($relativePath, $suffix)) {
|
||||||
|
$matchesSuffix = true;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $matchesSuffix) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($this->directories as $dir) {
|
||||||
|
if ($dir === '') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (str_starts_with($relativePath, $dir.'/')) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function toRelative(string $value, string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
$value = trim($value);
|
||||||
|
|
||||||
|
if ($value === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$real = @realpath($value);
|
||||||
|
$resolved = $real === false ? $value : $real;
|
||||||
|
|
||||||
|
$resolved = str_replace(DIRECTORY_SEPARATOR, '/', $resolved);
|
||||||
|
$root = str_replace(DIRECTORY_SEPARATOR, '/', rtrim($projectRoot, '/\\')).'/';
|
||||||
|
|
||||||
|
if (! str_starts_with($resolved.'/', $root)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rtrim(substr($resolved, strlen($root)), '/');
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function testSuiteFallback(string $projectRoot): ?string
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$testPath = TestSuite::getInstance()->testPath;
|
||||||
|
} catch (Throwable) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$real = @realpath($testPath);
|
||||||
|
$resolved = $real === false ? $testPath : $real;
|
||||||
|
$resolved = str_replace(DIRECTORY_SEPARATOR, '/', $resolved);
|
||||||
|
$root = str_replace(DIRECTORY_SEPARATOR, '/', rtrim($projectRoot, '/\\')).'/';
|
||||||
|
|
||||||
|
if (! str_starts_with($resolved.'/', $root)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rtrim(substr($resolved, strlen($root)), '/');
|
||||||
|
}
|
||||||
|
}
|
||||||
100
src/Plugins/Tia/WatchDefaults/Browser.php
Normal file
100
src/Plugins/Tia/WatchDefaults/Browser.php
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\WatchDefaults;
|
||||||
|
|
||||||
|
use Composer\InstalledVersions;
|
||||||
|
use Pest\Browser\Support\BrowserTestIdentifier;
|
||||||
|
use Pest\Factories\TestCaseFactory;
|
||||||
|
use Pest\Plugins\Tia\Contracts\WatchDefault;
|
||||||
|
use Pest\TestSuite;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Browser implements WatchDefault
|
||||||
|
{
|
||||||
|
public function applicable(): bool
|
||||||
|
{
|
||||||
|
return class_exists(InstalledVersions::class)
|
||||||
|
&& InstalledVersions::isInstalled('pestphp/pest-plugin-browser');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function defaults(string $projectRoot, string $testPath): array
|
||||||
|
{
|
||||||
|
$browserTargets = self::detectBrowserTestTargets($projectRoot, $testPath);
|
||||||
|
|
||||||
|
$globs = [
|
||||||
|
'resources/js/** !*.php',
|
||||||
|
'resources/css/** !*.php',
|
||||||
|
'public/hot !*.php',
|
||||||
|
'public/** !*.php',
|
||||||
|
];
|
||||||
|
|
||||||
|
$patterns = [];
|
||||||
|
|
||||||
|
foreach ($globs as $glob) {
|
||||||
|
$patterns[$glob] = $browserTargets;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $patterns;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
public static function detectBrowserTestTargets(string $projectRoot, string $testPath): array
|
||||||
|
{
|
||||||
|
$targets = [];
|
||||||
|
|
||||||
|
$candidate = $testPath.'/Browser';
|
||||||
|
|
||||||
|
if (is_dir($projectRoot.DIRECTORY_SEPARATOR.$candidate)) {
|
||||||
|
$targets[] = $candidate;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (class_exists(BrowserTestIdentifier::class)) {
|
||||||
|
$repo = TestSuite::getInstance()->tests;
|
||||||
|
|
||||||
|
foreach ($repo->getFilenames() as $filename) {
|
||||||
|
$factory = $repo->get($filename);
|
||||||
|
|
||||||
|
if (! $factory instanceof TestCaseFactory) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($factory->methods as $method) {
|
||||||
|
if (BrowserTestIdentifier::isBrowserTest($method)) {
|
||||||
|
$rel = self::fileRelative($projectRoot, $filename);
|
||||||
|
|
||||||
|
if ($rel !== null) {
|
||||||
|
$targets[] = $rel;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values(array_unique($targets));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function fileRelative(string $projectRoot, string $path): ?string
|
||||||
|
{
|
||||||
|
$real = @realpath($path);
|
||||||
|
|
||||||
|
if ($real === false) {
|
||||||
|
$real = $path;
|
||||||
|
}
|
||||||
|
|
||||||
|
$root = rtrim($projectRoot, DIRECTORY_SEPARATOR).DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
|
if (! str_starts_with($real, $root)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return str_replace(DIRECTORY_SEPARATOR, '/', substr($real, strlen($root)));
|
||||||
|
}
|
||||||
|
}
|
||||||
28
src/Plugins/Tia/WatchDefaults/Inertia.php
Normal file
28
src/Plugins/Tia/WatchDefaults/Inertia.php
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\WatchDefaults;
|
||||||
|
|
||||||
|
use Composer\InstalledVersions;
|
||||||
|
use Pest\Plugins\Tia\Contracts\WatchDefault;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Inertia implements WatchDefault
|
||||||
|
{
|
||||||
|
public function applicable(): bool
|
||||||
|
{
|
||||||
|
return class_exists(InstalledVersions::class)
|
||||||
|
&& (InstalledVersions::isInstalled('inertiajs/inertia-laravel')
|
||||||
|
|| InstalledVersions::isInstalled('rompetomp/inertia-bundle'));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function defaults(string $projectRoot, string $testPath): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'resources/js/** !*.php' => [$testPath],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
41
src/Plugins/Tia/WatchDefaults/Laravel.php
Normal file
41
src/Plugins/Tia/WatchDefaults/Laravel.php
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\WatchDefaults;
|
||||||
|
|
||||||
|
use Composer\InstalledVersions;
|
||||||
|
use Pest\Plugins\Tia\Contracts\WatchDefault;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Laravel implements WatchDefault
|
||||||
|
{
|
||||||
|
public function applicable(): bool
|
||||||
|
{
|
||||||
|
return class_exists(InstalledVersions::class)
|
||||||
|
&& InstalledVersions::isInstalled('laravel/framework');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function defaults(string $projectRoot, string $testPath): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'database/migrations/**/*.php' => [$testPath],
|
||||||
|
|
||||||
|
'storage/fixtures/**/*' => [$testPath],
|
||||||
|
|
||||||
|
'app/** !*.php' => [$testPath],
|
||||||
|
|
||||||
|
'resources/views/**' => [$testPath],
|
||||||
|
|
||||||
|
'lang/**' => [$testPath],
|
||||||
|
'resources/lang/**' => [$testPath],
|
||||||
|
|
||||||
|
'vite.config.* !*.php' => [$testPath],
|
||||||
|
'webpack.mix.* !*.php' => [$testPath],
|
||||||
|
'tailwind.config.* !*.php' => [$testPath],
|
||||||
|
'postcss.config.* !*.php' => [$testPath],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
32
src/Plugins/Tia/WatchDefaults/Livewire.php
Normal file
32
src/Plugins/Tia/WatchDefaults/Livewire.php
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\WatchDefaults;
|
||||||
|
|
||||||
|
use Composer\InstalledVersions;
|
||||||
|
use Pest\Plugins\Tia\Contracts\WatchDefault;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Livewire implements WatchDefault
|
||||||
|
{
|
||||||
|
public function applicable(): bool
|
||||||
|
{
|
||||||
|
return class_exists(InstalledVersions::class)
|
||||||
|
&& InstalledVersions::isInstalled('livewire/livewire');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function defaults(string $projectRoot, string $testPath): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'resources/views/livewire/**/*.blade.php' => [$testPath],
|
||||||
|
'resources/views/components/**/*.blade.php' => [$testPath],
|
||||||
|
'resources/views/pages/**/*.blade.php' => [$testPath],
|
||||||
|
|
||||||
|
'resources/js/**/*.js' => [$testPath],
|
||||||
|
'resources/js/**/*.ts' => [$testPath],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
38
src/Plugins/Tia/WatchDefaults/Php.php
Normal file
38
src/Plugins/Tia/WatchDefaults/Php.php
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\WatchDefaults;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Contracts\WatchDefault;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Php implements WatchDefault
|
||||||
|
{
|
||||||
|
public function applicable(): bool
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function defaults(string $projectRoot, string $testPath): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'.env' => [$testPath],
|
||||||
|
'.env.testing' => [$testPath],
|
||||||
|
'.env.local' => [$testPath],
|
||||||
|
'.env.*.local' => [$testPath],
|
||||||
|
|
||||||
|
'docker-compose.yml' => [$testPath],
|
||||||
|
'docker-compose.yaml' => [$testPath],
|
||||||
|
|
||||||
|
'phpunit.xml*' => [$testPath],
|
||||||
|
|
||||||
|
$testPath.'/Fixtures/**/*' => [$testPath],
|
||||||
|
$testPath.'/**/Fixtures/**/*' => [$testPath],
|
||||||
|
|
||||||
|
$testPath.'/.pest/snapshots/**/*.snap' => [$testPath],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
42
src/Plugins/Tia/WatchDefaults/Symfony.php
Normal file
42
src/Plugins/Tia/WatchDefaults/Symfony.php
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia\WatchDefaults;
|
||||||
|
|
||||||
|
use Composer\InstalledVersions;
|
||||||
|
use Pest\Plugins\Tia\Contracts\WatchDefault;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class Symfony implements WatchDefault
|
||||||
|
{
|
||||||
|
public function applicable(): bool
|
||||||
|
{
|
||||||
|
return class_exists(InstalledVersions::class)
|
||||||
|
&& InstalledVersions::isInstalled('symfony/framework-bundle');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function defaults(string $projectRoot, string $testPath): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'config/** !*.php' => [$testPath],
|
||||||
|
'config/routes/** !*.php' => [$testPath],
|
||||||
|
|
||||||
|
'migrations/**/*.php' => [$testPath],
|
||||||
|
'src/Migrations/**/*.php' => [$testPath],
|
||||||
|
|
||||||
|
'templates/** !*.php' => [$testPath],
|
||||||
|
|
||||||
|
'translations/** !*.php' => [$testPath],
|
||||||
|
|
||||||
|
'config/doctrine/**/*.xml' => [$testPath],
|
||||||
|
'config/doctrine/**/*.yaml' => [$testPath],
|
||||||
|
|
||||||
|
'webpack.config.js' => [$testPath],
|
||||||
|
'importmap.php' => [$testPath],
|
||||||
|
'assets/** !*.php' => [$testPath],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
331
src/Plugins/Tia/WatchPatterns.php
Normal file
331
src/Plugins/Tia/WatchPatterns.php
Normal file
@ -0,0 +1,331 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Contracts\WatchDefault;
|
||||||
|
use Pest\TestSuite;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class WatchPatterns
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var array<int, class-string<WatchDefault>>
|
||||||
|
*/
|
||||||
|
private const array DEFAULTS = [
|
||||||
|
WatchDefaults\Php::class,
|
||||||
|
WatchDefaults\Laravel::class,
|
||||||
|
WatchDefaults\Symfony::class,
|
||||||
|
WatchDefaults\Livewire::class,
|
||||||
|
WatchDefaults\Inertia::class,
|
||||||
|
WatchDefaults\Browser::class,
|
||||||
|
];
|
||||||
|
|
||||||
|
private const array VCS_DIRS = ['.git', '.svn', '.hg'];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<string, array<int, string>> raw pattern key → list of project-relative test dirs/files
|
||||||
|
*/
|
||||||
|
private array $patterns = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array<string, array{include: string, excludes: array<int, string>, allowDotfiles: bool}>
|
||||||
|
*/
|
||||||
|
private array $parsed = [];
|
||||||
|
|
||||||
|
private bool $enabled = false;
|
||||||
|
|
||||||
|
private bool $locally = false;
|
||||||
|
|
||||||
|
private bool $filtered = false;
|
||||||
|
|
||||||
|
private bool $baselined = false;
|
||||||
|
|
||||||
|
public function useDefaults(string $projectRoot): void
|
||||||
|
{
|
||||||
|
$testPath = TestSuite::getInstance()->testPath;
|
||||||
|
|
||||||
|
foreach (self::DEFAULTS as $class) {
|
||||||
|
$default = new $class;
|
||||||
|
|
||||||
|
if (! $default->applicable()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($default->defaults($projectRoot, $testPath) as $key => $dirs) {
|
||||||
|
$this->patterns[$key] = array_values(array_unique(
|
||||||
|
array_merge($this->patterns[$key] ?? [], $dirs),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, string> $patterns pattern key → project-relative test dir/file
|
||||||
|
*/
|
||||||
|
public function add(array $patterns): void
|
||||||
|
{
|
||||||
|
foreach ($patterns as $key => $dir) {
|
||||||
|
$this->patterns[$key] = array_values(array_unique(
|
||||||
|
array_merge($this->patterns[$key] ?? [], [$dir]),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $projectRoot Absolute path.
|
||||||
|
* @param array<int, string> $changedFiles Project-relative paths.
|
||||||
|
* @return array<int, string> Project-relative test dirs/files.
|
||||||
|
*/
|
||||||
|
public function matchedDirectories(string $projectRoot, array $changedFiles): array
|
||||||
|
{
|
||||||
|
if ($this->patterns === []) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$matched = [];
|
||||||
|
|
||||||
|
foreach ($changedFiles as $file) {
|
||||||
|
foreach ($this->patterns as $key => $dirs) {
|
||||||
|
if (! $this->keyMatches($key, $file)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($dirs as $dir) {
|
||||||
|
$matched[$dir] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_keys($matched);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $directories Project-relative dirs/files.
|
||||||
|
* @param array<int, string> $allTestFiles Project-relative test files from graph.
|
||||||
|
* @return array<int, string>
|
||||||
|
*/
|
||||||
|
public function testsUnderDirectories(array $directories, array $allTestFiles): array
|
||||||
|
{
|
||||||
|
if ($directories === []) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$affected = [];
|
||||||
|
|
||||||
|
foreach ($allTestFiles as $testFile) {
|
||||||
|
foreach ($directories as $target) {
|
||||||
|
if ($testFile === $target) {
|
||||||
|
$affected[] = $testFile;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
$prefix = rtrim($target, '/').'/';
|
||||||
|
|
||||||
|
if (str_starts_with($testFile, $prefix)) {
|
||||||
|
$affected[] = $testFile;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $affected;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function markEnabled(): void
|
||||||
|
{
|
||||||
|
$this->enabled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isEnabled(): bool
|
||||||
|
{
|
||||||
|
return $this->enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function markLocally(): void
|
||||||
|
{
|
||||||
|
$this->locally = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isLocally(): bool
|
||||||
|
{
|
||||||
|
return $this->locally;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function markFiltered(): void
|
||||||
|
{
|
||||||
|
$this->filtered = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isFiltered(): bool
|
||||||
|
{
|
||||||
|
return $this->filtered;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function markBaselined(): void
|
||||||
|
{
|
||||||
|
$this->baselined = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isBaselined(): bool
|
||||||
|
{
|
||||||
|
return $this->baselined;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function reset(): void
|
||||||
|
{
|
||||||
|
$this->patterns = [];
|
||||||
|
$this->parsed = [];
|
||||||
|
$this->enabled = false;
|
||||||
|
$this->locally = false;
|
||||||
|
$this->filtered = false;
|
||||||
|
$this->baselined = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function keyMatches(string $key, string $file): bool
|
||||||
|
{
|
||||||
|
$rule = $this->parse($key);
|
||||||
|
|
||||||
|
if (! $this->globMatches($rule['include'], $file)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$file = str_replace('\\', '/', $file);
|
||||||
|
|
||||||
|
if ($this->touchesVcs($file)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $rule['allowDotfiles'] && $this->touchesDotfile($file)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($rule['excludes'] as $exclude) {
|
||||||
|
if ($this->excludeMatches($exclude, $file)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{include: string, excludes: array<int, string>, allowDotfiles: bool}
|
||||||
|
*/
|
||||||
|
private function parse(string $key): array
|
||||||
|
{
|
||||||
|
if (isset($this->parsed[$key])) {
|
||||||
|
return $this->parsed[$key];
|
||||||
|
}
|
||||||
|
|
||||||
|
$tokens = preg_split('/\s+/', trim($key)) ?: [];
|
||||||
|
|
||||||
|
$include = '';
|
||||||
|
$excludes = [];
|
||||||
|
|
||||||
|
foreach ($tokens as $token) {
|
||||||
|
if ($token === '') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($token[0] === '!') {
|
||||||
|
$excludes[] = substr($token, 1);
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($include === '') {
|
||||||
|
$include = $token;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->parsed[$key] = [
|
||||||
|
'include' => $include,
|
||||||
|
'excludes' => $excludes,
|
||||||
|
'allowDotfiles' => $this->patternTargetsDotfiles($include),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function patternTargetsDotfiles(string $pattern): bool
|
||||||
|
{
|
||||||
|
foreach (explode('/', str_replace('\\', '/', $pattern)) as $segment) {
|
||||||
|
if ($segment !== '' && $segment[0] === '.') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function touchesVcs(string $file): bool
|
||||||
|
{
|
||||||
|
foreach (explode('/', $file) as $segment) {
|
||||||
|
if (in_array($segment, self::VCS_DIRS, true)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function touchesDotfile(string $file): bool
|
||||||
|
{
|
||||||
|
foreach (explode('/', $file) as $segment) {
|
||||||
|
if ($segment !== '' && $segment[0] === '.') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function excludeMatches(string $exclude, string $file): bool
|
||||||
|
{
|
||||||
|
$pattern = str_contains($exclude, '/') ? $exclude : '**/'.$exclude;
|
||||||
|
|
||||||
|
if ($this->globMatches($pattern, $file)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->globMatches($exclude, basename($file));
|
||||||
|
}
|
||||||
|
|
||||||
|
private function globMatches(string $pattern, string $file): bool
|
||||||
|
{
|
||||||
|
$pattern = str_replace('\\', '/', $pattern);
|
||||||
|
$file = str_replace('\\', '/', $file);
|
||||||
|
|
||||||
|
$regex = '';
|
||||||
|
$len = strlen($pattern);
|
||||||
|
$i = 0;
|
||||||
|
|
||||||
|
while ($i < $len) {
|
||||||
|
$c = $pattern[$i];
|
||||||
|
|
||||||
|
if ($c === '*' && isset($pattern[$i + 1]) && $pattern[$i + 1] === '*') {
|
||||||
|
$regex .= '.*';
|
||||||
|
$i += 2;
|
||||||
|
|
||||||
|
if (isset($pattern[$i]) && $pattern[$i] === '/') {
|
||||||
|
$i++;
|
||||||
|
}
|
||||||
|
} elseif ($c === '*') {
|
||||||
|
$regex .= '[^/]*';
|
||||||
|
$i++;
|
||||||
|
} elseif ($c === '?') {
|
||||||
|
$regex .= '[^/]';
|
||||||
|
$i++;
|
||||||
|
} else {
|
||||||
|
$regex .= preg_quote($c, '#');
|
||||||
|
$i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (bool) preg_match('#^'.$regex.'$#', $file);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -59,8 +59,10 @@ final class SnapshotRepository
|
|||||||
{
|
{
|
||||||
$snapshotFilename = $this->getSnapshotFilename();
|
$snapshotFilename = $this->getSnapshotFilename();
|
||||||
|
|
||||||
if (! file_exists(dirname($snapshotFilename))) {
|
$directory = dirname($snapshotFilename);
|
||||||
mkdir(dirname($snapshotFilename), 0755, true);
|
|
||||||
|
if (! is_dir($directory)) {
|
||||||
|
@mkdir($directory, 0755, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
file_put_contents($snapshotFilename, $snapshot);
|
file_put_contents($snapshotFilename, $snapshot);
|
||||||
|
|||||||
95
src/Restarters/PcovRestarter.php
Normal file
95
src/Restarters/PcovRestarter.php
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Restarters;
|
||||||
|
|
||||||
|
use Pest\Contracts\Restarter;
|
||||||
|
use Pest\Plugins\Tia;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class PcovRestarter implements Restarter
|
||||||
|
{
|
||||||
|
private const string ENV_RESTARTED = 'PEST_PCOV_RESTARTER_RESTARTED';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $arguments
|
||||||
|
*/
|
||||||
|
public function maybeRestart(string $projectRoot, array $arguments): void
|
||||||
|
{
|
||||||
|
if (! extension_loaded('pcov')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (getenv(self::ENV_RESTARTED) === '1') {
|
||||||
|
putenv(self::ENV_RESTARTED);
|
||||||
|
unset($_ENV[self::ENV_RESTARTED]);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! Tia::isEnabledForRun($arguments)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$desired = $this->normalise($projectRoot);
|
||||||
|
$current = $this->normalise((string) ini_get('pcov.directory'));
|
||||||
|
|
||||||
|
if ($current === $desired) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->restart($projectRoot, $arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $arguments
|
||||||
|
*/
|
||||||
|
private function restart(string $projectRoot, array $arguments): void
|
||||||
|
{
|
||||||
|
$env = $this->inheritEnv();
|
||||||
|
$env[self::ENV_RESTARTED] = '1';
|
||||||
|
|
||||||
|
$command = array_merge(
|
||||||
|
[PHP_BINARY, '-d', 'pcov.directory='.$projectRoot],
|
||||||
|
array_values($arguments),
|
||||||
|
);
|
||||||
|
|
||||||
|
$proc = @proc_open(
|
||||||
|
$command,
|
||||||
|
[STDIN, STDOUT, STDERR],
|
||||||
|
$pipes,
|
||||||
|
null,
|
||||||
|
$env,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (! is_resource($proc)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$exitCode = proc_close($proc);
|
||||||
|
|
||||||
|
exit($exitCode === -1 ? 1 : $exitCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string>
|
||||||
|
*/
|
||||||
|
private function inheritEnv(): array
|
||||||
|
{
|
||||||
|
$env = [];
|
||||||
|
|
||||||
|
foreach (getenv() as $name => $value) {
|
||||||
|
$env[$name] = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $env;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function normalise(string $path): string
|
||||||
|
{
|
||||||
|
return rtrim($path, '/\\');
|
||||||
|
}
|
||||||
|
}
|
||||||
113
src/Restarters/XdebugRestarter.php
Normal file
113
src/Restarters/XdebugRestarter.php
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Restarters;
|
||||||
|
|
||||||
|
use Composer\XdebugHandler\XdebugHandler;
|
||||||
|
use Pest\Contracts\Restarter;
|
||||||
|
use Pest\Plugins\Tia;
|
||||||
|
use Pest\Plugins\Tia\Fingerprint;
|
||||||
|
use Pest\Plugins\Tia\Graph;
|
||||||
|
use Pest\Plugins\Tia\Storage;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class XdebugRestarter implements Restarter
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $arguments
|
||||||
|
*/
|
||||||
|
public function maybeRestart(string $projectRoot, array $arguments): void
|
||||||
|
{
|
||||||
|
if (! class_exists(XdebugHandler::class)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! extension_loaded('xdebug')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $this->xdebugIsCoverageOnly()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $this->runLooksDroppable($arguments, $projectRoot)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
(new XdebugHandler('pest'))->check();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function xdebugIsCoverageOnly(): bool
|
||||||
|
{
|
||||||
|
if (! function_exists('xdebug_info')) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$modes = @xdebug_info('mode');
|
||||||
|
|
||||||
|
if (! is_array($modes)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$modes = array_values(array_filter($modes, is_string(...)));
|
||||||
|
|
||||||
|
if ($modes === []) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $modes === ['coverage'];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<int, string> $arguments
|
||||||
|
*/
|
||||||
|
private function runLooksDroppable(array $arguments, string $projectRoot): bool
|
||||||
|
{
|
||||||
|
foreach ($arguments as $value) {
|
||||||
|
if ($value === '--coverage'
|
||||||
|
|| str_starts_with($value, '--coverage=')
|
||||||
|
|| str_starts_with($value, '--coverage-')) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($value === '--fresh') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! Tia::isEnabledForRun($arguments)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->tiaWillReplay($projectRoot);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function tiaWillReplay(string $projectRoot): bool
|
||||||
|
{
|
||||||
|
$path = Storage::tempDir($projectRoot).DIRECTORY_SEPARATOR.Tia::KEY_GRAPH;
|
||||||
|
|
||||||
|
if (! is_file($path)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$json = @file_get_contents($path);
|
||||||
|
|
||||||
|
if ($json === false) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$graph = Graph::decode($json, $projectRoot);
|
||||||
|
|
||||||
|
if (! $graph instanceof Graph) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Fingerprint::structuralMatches(
|
||||||
|
$graph->fingerprint(),
|
||||||
|
Fingerprint::compute($projectRoot),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureShardTimingFinished.php
Normal file
22
src/Subscribers/EnsureShardTimingFinished.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use PHPUnit\Event\TestSuite\Finished;
|
||||||
|
use PHPUnit\Event\TestSuite\FinishedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class EnsureShardTimingFinished implements FinishedSubscriber
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Runs the subscriber.
|
||||||
|
*/
|
||||||
|
public function notify(Finished $event): void
|
||||||
|
{
|
||||||
|
EnsureShardTimingsAreCollected::finished($event);
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureShardTimingStarted.php
Normal file
22
src/Subscribers/EnsureShardTimingStarted.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use PHPUnit\Event\TestSuite\Started;
|
||||||
|
use PHPUnit\Event\TestSuite\StartedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class EnsureShardTimingStarted implements StartedSubscriber
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Runs the subscriber.
|
||||||
|
*/
|
||||||
|
public function notify(Started $event): void
|
||||||
|
{
|
||||||
|
EnsureShardTimingsAreCollected::started($event);
|
||||||
|
}
|
||||||
|
}
|
||||||
75
src/Subscribers/EnsureShardTimingsAreCollected.php
Normal file
75
src/Subscribers/EnsureShardTimingsAreCollected.php
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use PHPUnit\Event\Telemetry\HRTime;
|
||||||
|
use PHPUnit\Event\TestSuite\Finished;
|
||||||
|
use PHPUnit\Event\TestSuite\Started;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class EnsureShardTimingsAreCollected
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* The start times for each test class.
|
||||||
|
*
|
||||||
|
* @var array<string, HRTime>
|
||||||
|
*/
|
||||||
|
private static array $startTimes = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The collected timings for each test class.
|
||||||
|
*
|
||||||
|
* @var array<string, float>
|
||||||
|
*/
|
||||||
|
private static array $timings = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Records the start time for a test suite.
|
||||||
|
*/
|
||||||
|
public static function started(Started $event): void
|
||||||
|
{
|
||||||
|
if (! $event->testSuite()->isForTestClass()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$name = preg_replace('/^P\\\\/', '', $event->testSuite()->name());
|
||||||
|
|
||||||
|
if (is_string($name)) {
|
||||||
|
self::$startTimes[$name] = $event->telemetryInfo()->time();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Records the duration for a test suite.
|
||||||
|
*/
|
||||||
|
public static function finished(Finished $event): void
|
||||||
|
{
|
||||||
|
if (! $event->testSuite()->isForTestClass()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$name = preg_replace('/^P\\\\/', '', $event->testSuite()->name());
|
||||||
|
|
||||||
|
if (! is_string($name) || ! isset(self::$startTimes[$name])) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$duration = $event->telemetryInfo()->time()->duration(self::$startTimes[$name]);
|
||||||
|
|
||||||
|
self::$timings[$name] = round($duration->asFloat(), 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the collected timings.
|
||||||
|
*
|
||||||
|
* @return array<string, float>
|
||||||
|
*/
|
||||||
|
public static function timings(): array
|
||||||
|
{
|
||||||
|
return self::$timings;
|
||||||
|
}
|
||||||
|
}
|
||||||
32
src/Subscribers/EnsureTiaAssertionsAreRecordedOnFinished.php
Normal file
32
src/Subscribers/EnsureTiaAssertionsAreRecordedOnFinished.php
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Code\TestMethod;
|
||||||
|
use PHPUnit\Event\Test\Finished;
|
||||||
|
use PHPUnit\Event\Test\FinishedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaAssertionsAreRecordedOnFinished implements FinishedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(Finished $event): void
|
||||||
|
{
|
||||||
|
$test = $event->test();
|
||||||
|
|
||||||
|
if ($test instanceof TestMethod) {
|
||||||
|
$this->collector->recordAssertions(
|
||||||
|
$test->className().'::'.$test->methodName(),
|
||||||
|
$event->numberOfAssertionsPerformed(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->collector->finishTest();
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureTiaEnds.php
Normal file
22
src/Subscribers/EnsureTiaEnds.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Recorder;
|
||||||
|
use PHPUnit\Event\Test\Finished;
|
||||||
|
use PHPUnit\Event\Test\FinishedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaEnds implements FinishedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private Recorder $recorder) {}
|
||||||
|
|
||||||
|
public function notify(Finished $event): void
|
||||||
|
{
|
||||||
|
$this->recorder->endTest();
|
||||||
|
}
|
||||||
|
}
|
||||||
45
src/Subscribers/EnsureTiaIsRunningPestTestsOnly.php
Normal file
45
src/Subscribers/EnsureTiaIsRunningPestTestsOnly.php
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Exceptions\TiaRequiresPestTests;
|
||||||
|
use Pest\Panic;
|
||||||
|
use Pest\Plugins\Tia\Recorder;
|
||||||
|
use PHPUnit\Event\Code\TestMethod;
|
||||||
|
use PHPUnit\Event\Test\Prepared;
|
||||||
|
use PHPUnit\Event\Test\PreparedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaIsRunningPestTestsOnly implements PreparedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private Recorder $recorder) {}
|
||||||
|
|
||||||
|
public function notify(Prepared $event): void
|
||||||
|
{
|
||||||
|
if (! $this->recorder->isActive()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$test = $event->test();
|
||||||
|
|
||||||
|
if (! $test instanceof TestMethod) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$className = $test->className();
|
||||||
|
|
||||||
|
if (! class_exists($className, false)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (method_exists($className, '__initializeTestCase')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Panic::with(new TiaRequiresPestTests($className, $test->file()));
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureTiaResultIsRecordedOnErrored.php
Normal file
22
src/Subscribers/EnsureTiaResultIsRecordedOnErrored.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Test\Errored;
|
||||||
|
use PHPUnit\Event\Test\ErroredSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaResultIsRecordedOnErrored implements ErroredSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(Errored $event): void
|
||||||
|
{
|
||||||
|
$this->collector->testErrored($event->throwable()->message());
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureTiaResultIsRecordedOnFailed.php
Normal file
22
src/Subscribers/EnsureTiaResultIsRecordedOnFailed.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Test\Failed;
|
||||||
|
use PHPUnit\Event\Test\FailedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaResultIsRecordedOnFailed implements FailedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(Failed $event): void
|
||||||
|
{
|
||||||
|
$this->collector->testFailed($event->throwable()->message());
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureTiaResultIsRecordedOnIncomplete.php
Normal file
22
src/Subscribers/EnsureTiaResultIsRecordedOnIncomplete.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Test\MarkedIncomplete;
|
||||||
|
use PHPUnit\Event\Test\MarkedIncompleteSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaResultIsRecordedOnIncomplete implements MarkedIncompleteSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(MarkedIncomplete $event): void
|
||||||
|
{
|
||||||
|
$this->collector->testIncomplete($event->throwable()->message());
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureTiaResultIsRecordedOnPassed.php
Normal file
22
src/Subscribers/EnsureTiaResultIsRecordedOnPassed.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Test\Passed;
|
||||||
|
use PHPUnit\Event\Test\PassedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaResultIsRecordedOnPassed implements PassedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(Passed $event): void
|
||||||
|
{
|
||||||
|
$this->collector->testPassed();
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureTiaResultIsRecordedOnRisky.php
Normal file
22
src/Subscribers/EnsureTiaResultIsRecordedOnRisky.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Test\ConsideredRisky;
|
||||||
|
use PHPUnit\Event\Test\ConsideredRiskySubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaResultIsRecordedOnRisky implements ConsideredRiskySubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(ConsideredRisky $event): void
|
||||||
|
{
|
||||||
|
$this->collector->testRisky($event->message());
|
||||||
|
}
|
||||||
|
}
|
||||||
22
src/Subscribers/EnsureTiaResultIsRecordedOnSkipped.php
Normal file
22
src/Subscribers/EnsureTiaResultIsRecordedOnSkipped.php
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Test\Skipped;
|
||||||
|
use PHPUnit\Event\Test\SkippedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaResultIsRecordedOnSkipped implements SkippedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(Skipped $event): void
|
||||||
|
{
|
||||||
|
$this->collector->testSkipped($event->message());
|
||||||
|
}
|
||||||
|
}
|
||||||
27
src/Subscribers/EnsureTiaResultsAreCollected.php
Normal file
27
src/Subscribers/EnsureTiaResultsAreCollected.php
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ResultCollector;
|
||||||
|
use PHPUnit\Event\Code\TestMethod;
|
||||||
|
use PHPUnit\Event\Test\PreparationStarted;
|
||||||
|
use PHPUnit\Event\Test\PreparationStartedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaResultsAreCollected implements PreparationStartedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private ResultCollector $collector) {}
|
||||||
|
|
||||||
|
public function notify(PreparationStarted $event): void
|
||||||
|
{
|
||||||
|
$test = $event->test();
|
||||||
|
|
||||||
|
if ($test instanceof TestMethod) {
|
||||||
|
$this->collector->testPrepared($test->className().'::'.$test->methodName(), $test->file());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
33
src/Subscribers/EnsureTiaStarts.php
Normal file
33
src/Subscribers/EnsureTiaStarts.php
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Subscribers;
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\Recorder;
|
||||||
|
use PHPUnit\Event\Code\TestMethod;
|
||||||
|
use PHPUnit\Event\Test\Prepared;
|
||||||
|
use PHPUnit\Event\Test\PreparedSubscriber;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class EnsureTiaStarts implements PreparedSubscriber
|
||||||
|
{
|
||||||
|
public function __construct(private Recorder $recorder) {}
|
||||||
|
|
||||||
|
public function notify(Prepared $event): void
|
||||||
|
{
|
||||||
|
if (! $this->recorder->isActive()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$test = $event->test();
|
||||||
|
|
||||||
|
if (! $test instanceof TestMethod) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->recorder->beginTest($test->className(), $test->methodName(), $test->file());
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -5,6 +5,7 @@ declare(strict_types=1);
|
|||||||
namespace Pest\Support;
|
namespace Pest\Support;
|
||||||
|
|
||||||
use Pest\Exceptions\ShouldNotHappen;
|
use Pest\Exceptions\ShouldNotHappen;
|
||||||
|
use Pest\Plugins\Tia\CoverageMerger;
|
||||||
use SebastianBergmann\CodeCoverage\CodeCoverage;
|
use SebastianBergmann\CodeCoverage\CodeCoverage;
|
||||||
use SebastianBergmann\CodeCoverage\Node\Directory;
|
use SebastianBergmann\CodeCoverage\Node\Directory;
|
||||||
use SebastianBergmann\CodeCoverage\Node\File;
|
use SebastianBergmann\CodeCoverage\Node\File;
|
||||||
@ -88,6 +89,8 @@ final class Coverage
|
|||||||
throw ShouldNotHappen::fromMessage(sprintf('Coverage not found in path: %s.', $reportPath));
|
throw ShouldNotHappen::fromMessage(sprintf('Coverage not found in path: %s.', $reportPath));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CoverageMerger::applyIfMarked($reportPath);
|
||||||
|
|
||||||
/** @var CodeCoverage $codeCoverage */
|
/** @var CodeCoverage $codeCoverage */
|
||||||
$codeCoverage = require $reportPath;
|
$codeCoverage = require $reportPath;
|
||||||
unlink($reportPath);
|
unlink($reportPath);
|
||||||
|
|||||||
@ -11,6 +11,7 @@ use PHPUnit\Event\Code\TestDoxBuilder;
|
|||||||
use PHPUnit\Event\Code\TestMethod;
|
use PHPUnit\Event\Code\TestMethod;
|
||||||
use PHPUnit\Event\Code\ThrowableBuilder;
|
use PHPUnit\Event\Code\ThrowableBuilder;
|
||||||
use PHPUnit\Event\Test\Errored;
|
use PHPUnit\Event\Test\Errored;
|
||||||
|
use PHPUnit\Event\Test\Failed;
|
||||||
use PHPUnit\Event\Test\PhpunitDeprecationTriggered;
|
use PHPUnit\Event\Test\PhpunitDeprecationTriggered;
|
||||||
use PHPUnit\Event\Test\PhpunitErrorTriggered;
|
use PHPUnit\Event\Test\PhpunitErrorTriggered;
|
||||||
use PHPUnit\Event\Test\PhpunitNoticeTriggered;
|
use PHPUnit\Event\Test\PhpunitNoticeTriggered;
|
||||||
@ -40,11 +41,16 @@ final class StateGenerator
|
|||||||
}
|
}
|
||||||
|
|
||||||
foreach ($testResult->testFailedEvents() as $testResultEvent) {
|
foreach ($testResult->testFailedEvents() as $testResultEvent) {
|
||||||
|
if ($testResultEvent instanceof Failed) {
|
||||||
$state->add(TestResult::fromPestParallelTestCase(
|
$state->add(TestResult::fromPestParallelTestCase(
|
||||||
$testResultEvent->test(),
|
$testResultEvent->test(),
|
||||||
TestResult::FAIL,
|
TestResult::FAIL,
|
||||||
$testResultEvent->throwable()
|
$testResultEvent->throwable()
|
||||||
));
|
));
|
||||||
|
} else {
|
||||||
|
// @phpstan-ignore-next-line
|
||||||
|
$state->add(TestResult::fromBeforeFirstTestMethodErrored($testResultEvent));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->addTriggeredPhpunitEvents($state, $testResult->testTriggeredPhpunitErrorEvents(), TestResult::FAIL);
|
$this->addTriggeredPhpunitEvents($state, $testResult->testTriggeredPhpunitErrorEvents(), TestResult::FAIL);
|
||||||
|
|||||||
55
src/TestCaseFilters/TiaTestCaseFilter.php
Normal file
55
src/TestCaseFilters/TiaTestCaseFilter.php
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\TestCaseFilters;
|
||||||
|
|
||||||
|
use Pest\Contracts\TestCaseFilter;
|
||||||
|
use Pest\Plugins\Tia\Graph;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final readonly class TiaTestCaseFilter implements TestCaseFilter
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param array<string, true> $affectedTestFiles Keys are project-relative test file paths.
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
private string $projectRoot,
|
||||||
|
private Graph $graph,
|
||||||
|
private array $affectedTestFiles,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function accept(string $testCaseFilename): bool
|
||||||
|
{
|
||||||
|
$rel = $this->relative($testCaseFilename);
|
||||||
|
|
||||||
|
if ($rel === null) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! $this->graph->knowsTest($rel)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return isset($this->affectedTestFiles[$rel]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function relative(string $path): ?string
|
||||||
|
{
|
||||||
|
$real = @realpath($path);
|
||||||
|
|
||||||
|
if ($real === false) {
|
||||||
|
$real = $path;
|
||||||
|
}
|
||||||
|
|
||||||
|
$root = rtrim($this->projectRoot, DIRECTORY_SEPARATOR).DIRECTORY_SEPARATOR;
|
||||||
|
|
||||||
|
if (! str_starts_with($real, $root)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return str_replace(DIRECTORY_SEPARATOR, '/', substr($real, strlen($root)));
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,7 +0,0 @@
|
|||||||
<div class="container">
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-md-12">
|
|
||||||
<h1>Snapshot</h1>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@ -1,7 +0,0 @@
|
|||||||
<div class="container">
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-md-12">
|
|
||||||
<h1>Snapshot</h1>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@ -1,5 +1,5 @@
|
|||||||
|
|
||||||
Pest Testing Framework 4.4.4.
|
Pest Testing Framework 4.7.1.
|
||||||
|
|
||||||
USAGE: pest <file> [options]
|
USAGE: pest <file> [options]
|
||||||
|
|
||||||
@ -49,6 +49,7 @@
|
|||||||
EXECUTION OPTIONS:
|
EXECUTION OPTIONS:
|
||||||
--parallel ........................................... Run tests in parallel
|
--parallel ........................................... Run tests in parallel
|
||||||
--update-snapshots Update snapshots for tests using the "toMatchSnapshot" expectation
|
--update-snapshots Update snapshots for tests using the "toMatchSnapshot" expectation
|
||||||
|
--update-shards Update shards.json with test timing data for time-balanced sharding
|
||||||
--globals-backup ................. Backup and restore $GLOBALS for each test
|
--globals-backup ................. Backup and restore $GLOBALS for each test
|
||||||
--static-backup ......... Backup and restore static properties for each test
|
--static-backup ......... Backup and restore static properties for each test
|
||||||
--strict-coverage ................... Be strict about code coverage metadata
|
--strict-coverage ................... Be strict about code coverage metadata
|
||||||
@ -90,7 +91,11 @@
|
|||||||
--cache-result ............................ Write test results to cache file
|
--cache-result ............................ Write test results to cache file
|
||||||
--do-not-cache-result .............. Do not write test results to cache file
|
--do-not-cache-result .............. Do not write test results to cache file
|
||||||
--order-by [order] Run tests in order: default|defects|depends|duration|no-depends|random|reverse|size
|
--order-by [order] Run tests in order: default|defects|depends|duration|no-depends|random|reverse|size
|
||||||
|
--resolve-dependencies ...................... Alias for "--order-by depends"
|
||||||
|
--ignore-dependencies .................... Alias for "--order-by no-depends"
|
||||||
|
--random-order ............................... Alias for "--order-by random"
|
||||||
--random-order-seed [N] Use the specified random seed when running tests in random order
|
--random-order-seed [N] Use the specified random seed when running tests in random order
|
||||||
|
--reverse-order ............................. Alias for "--order-by reverse"
|
||||||
|
|
||||||
REPORTING OPTIONS:
|
REPORTING OPTIONS:
|
||||||
--colors=[flag] ......... Use colors in output ("never", "auto" or "always")
|
--colors=[flag] ......... Use colors in output ("never", "auto" or "always")
|
||||||
|
|||||||
@ -1,3 +1,3 @@
|
|||||||
|
|
||||||
Pest Testing Framework 4.4.4.
|
Pest Testing Framework 4.7.1.
|
||||||
|
|
||||||
|
|||||||
@ -1,28 +1,56 @@
|
|||||||
##teamcity[testSuiteStarted name='Tests/tests/Failure' locationHint='pest_qn://tests/.tests/Failure.php' flowId='1234']
|
##teamcity[testSuiteStarted name='Tests/tests/Failure' locationHint='pest_qn://tests/.tests/Failure.php' flowId='1234']
|
||||||
##teamcity[testCount count='8' flowId='1234']
|
##teamcity[testCount count='8' flowId='1234']
|
||||||
|
##teamcity[testSuiteStarted name='Tests/tests/Failure' locationHint='pest_qn://tests/.tests/Failure.php' flowId='1234']
|
||||||
|
##teamcity[testCount count='8' flowId='1234']
|
||||||
##teamcity[testStarted name='it can fail with comparison' locationHint='pest_qn://tests/.tests/Failure.php::it can fail with comparison' flowId='1234']
|
##teamcity[testStarted name='it can fail with comparison' locationHint='pest_qn://tests/.tests/Failure.php::it can fail with comparison' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it can fail with comparison' locationHint='pest_qn://tests/.tests/Failure.php::it can fail with comparison' flowId='1234']
|
||||||
|
##teamcity[testFailed name='it can fail with comparison' message='Failed asserting that true matches expected false.' details='at tests/.tests/Failure.php:6' type='comparisonFailure' actual='true' expected='false' flowId='1234']
|
||||||
##teamcity[testFailed name='it can fail with comparison' message='Failed asserting that true matches expected false.' details='at tests/.tests/Failure.php:6' type='comparisonFailure' actual='true' expected='false' flowId='1234']
|
##teamcity[testFailed name='it can fail with comparison' message='Failed asserting that true matches expected false.' details='at tests/.tests/Failure.php:6' type='comparisonFailure' actual='true' expected='false' flowId='1234']
|
||||||
##teamcity[testFinished name='it can fail with comparison' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it can fail with comparison' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it can fail with comparison' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it can be ignored because of no assertions' locationHint='pest_qn://tests/.tests/Failure.php::it can be ignored because of no assertions' flowId='1234']
|
||||||
##teamcity[testStarted name='it can be ignored because of no assertions' locationHint='pest_qn://tests/.tests/Failure.php::it can be ignored because of no assertions' flowId='1234']
|
##teamcity[testStarted name='it can be ignored because of no assertions' locationHint='pest_qn://tests/.tests/Failure.php::it can be ignored because of no assertions' flowId='1234']
|
||||||
##teamcity[testIgnored name='it can be ignored because of no assertions' message='This test did not perform any assertions' details='' flowId='1234']
|
##teamcity[testIgnored name='it can be ignored because of no assertions' message='This test did not perform any assertions' details='' flowId='1234']
|
||||||
|
##teamcity[testIgnored name='it can be ignored because of no assertions' message='This test did not perform any assertions' details='' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it can be ignored because of no assertions' duration='100000' flowId='1234']
|
||||||
##teamcity[testFinished name='it can be ignored because of no assertions' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it can be ignored because of no assertions' duration='100000' flowId='1234']
|
||||||
##teamcity[testStarted name='it can be ignored because it is skipped' locationHint='pest_qn://tests/.tests/Failure.php::it can be ignored because it is skipped' flowId='1234']
|
##teamcity[testStarted name='it can be ignored because it is skipped' locationHint='pest_qn://tests/.tests/Failure.php::it can be ignored because it is skipped' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it can be ignored because it is skipped' locationHint='pest_qn://tests/.tests/Failure.php::it can be ignored because it is skipped' flowId='1234']
|
||||||
|
##teamcity[testIgnored name='it can be ignored because it is skipped' message='This test was ignored.' details='' flowId='1234']
|
||||||
##teamcity[testIgnored name='it can be ignored because it is skipped' message='This test was ignored.' details='' flowId='1234']
|
##teamcity[testIgnored name='it can be ignored because it is skipped' message='This test was ignored.' details='' flowId='1234']
|
||||||
##teamcity[testFinished name='it can be ignored because it is skipped' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it can be ignored because it is skipped' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it can be ignored because it is skipped' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it can fail' locationHint='pest_qn://tests/.tests/Failure.php::it can fail' flowId='1234']
|
||||||
##teamcity[testStarted name='it can fail' locationHint='pest_qn://tests/.tests/Failure.php::it can fail' flowId='1234']
|
##teamcity[testStarted name='it can fail' locationHint='pest_qn://tests/.tests/Failure.php::it can fail' flowId='1234']
|
||||||
##teamcity[testFailed name='it can fail' message='oh noo' details='at tests/.tests/Failure.php:18' flowId='1234']
|
##teamcity[testFailed name='it can fail' message='oh noo' details='at tests/.tests/Failure.php:18' flowId='1234']
|
||||||
|
##teamcity[testFailed name='it can fail' message='oh noo' details='at tests/.tests/Failure.php:18' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it can fail' duration='100000' flowId='1234']
|
||||||
##teamcity[testFinished name='it can fail' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it can fail' duration='100000' flowId='1234']
|
||||||
##teamcity[testStarted name='it throws exception' locationHint='pest_qn://tests/.tests/Failure.php::it throws exception' flowId='1234']
|
##teamcity[testStarted name='it throws exception' locationHint='pest_qn://tests/.tests/Failure.php::it throws exception' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it throws exception' locationHint='pest_qn://tests/.tests/Failure.php::it throws exception' flowId='1234']
|
||||||
|
##teamcity[testFailed name='it throws exception' message='Exception: test error' details='at tests/.tests/Failure.php:22' flowId='1234']
|
||||||
##teamcity[testFailed name='it throws exception' message='Exception: test error' details='at tests/.tests/Failure.php:22' flowId='1234']
|
##teamcity[testFailed name='it throws exception' message='Exception: test error' details='at tests/.tests/Failure.php:22' flowId='1234']
|
||||||
##teamcity[testFinished name='it throws exception' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it throws exception' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it throws exception' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it is not done yet' locationHint='pest_qn://tests/.tests/Failure.php::it is not done yet' flowId='1234']
|
||||||
##teamcity[testStarted name='it is not done yet' locationHint='pest_qn://tests/.tests/Failure.php::it is not done yet' flowId='1234']
|
##teamcity[testStarted name='it is not done yet' locationHint='pest_qn://tests/.tests/Failure.php::it is not done yet' flowId='1234']
|
||||||
##teamcity[testFinished name='it is not done yet' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it is not done yet' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it is not done yet' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testStarted name='build this one.' locationHint='pest_qn://tests/.tests/Failure.php::build this one.' flowId='1234']
|
||||||
##teamcity[testStarted name='build this one.' locationHint='pest_qn://tests/.tests/Failure.php::build this one.' flowId='1234']
|
##teamcity[testStarted name='build this one.' locationHint='pest_qn://tests/.tests/Failure.php::build this one.' flowId='1234']
|
||||||
##teamcity[testFinished name='build this one.' duration='100000' flowId='1234']
|
##teamcity[testFinished name='build this one.' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='build this one.' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it is passing' locationHint='pest_qn://tests/.tests/Failure.php::it is passing' flowId='1234']
|
||||||
##teamcity[testStarted name='it is passing' locationHint='pest_qn://tests/.tests/Failure.php::it is passing' flowId='1234']
|
##teamcity[testStarted name='it is passing' locationHint='pest_qn://tests/.tests/Failure.php::it is passing' flowId='1234']
|
||||||
##teamcity[testFinished name='it is passing' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it is passing' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it is passing' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testSuiteFinished name='Tests/tests/Failure' flowId='1234']
|
||||||
##teamcity[testSuiteFinished name='Tests/tests/Failure' flowId='1234']
|
##teamcity[testSuiteFinished name='Tests/tests/Failure' flowId='1234']
|
||||||
|
|
||||||
[90mTests:[39m [31;1m3 failed[39;22m[90m,[39m[39m [39m[33;1m1 risky[39;22m[90m,[39m[39m [39m[36;1m2 todos[39;22m[90m,[39m[39m [39m[33;1m1 skipped[39;22m[90m,[39m[39m [39m[32;1m1 passed[39;22m[90m (3 assertions)[39m
|
[90mTests:[39m [31;1m3 failed[39;22m[90m,[39m[39m [39m[33;1m1 risky[39;22m[90m,[39m[39m [39m[36;1m2 todos[39;22m[90m,[39m[39m [39m[33;1m1 skipped[39;22m[90m,[39m[39m [39m[32;1m1 passed[39;22m[90m (3 assertions)[39m
|
||||||
[90mDuration:[39m [39m1.00s[39m
|
[90mDuration:[39m [39m1.00s[39m
|
||||||
|
|
||||||
|
|
||||||
|
[90mTests:[39m [31;1m3 failed[39;22m[90m,[39m[39m [39m[33;1m1 risky[39;22m[90m,[39m[39m [39m[36;1m2 todos[39;22m[90m,[39m[39m [39m[33;1m1 skipped[39;22m[90m,[39m[39m [39m[32;1m1 passed[39;22m[90m (3 assertions)[39m
|
||||||
|
[90mDuration:[39m [39m1.00s[39m
|
||||||
|
|
||||||
|
|||||||
@ -1,19 +1,38 @@
|
|||||||
##teamcity[testSuiteStarted name='Tests/tests/SuccessOnly' locationHint='pest_qn://tests/.tests/SuccessOnly.php' flowId='1234']
|
##teamcity[testSuiteStarted name='Tests/tests/SuccessOnly' locationHint='pest_qn://tests/.tests/SuccessOnly.php' flowId='1234']
|
||||||
##teamcity[testCount count='4' flowId='1234']
|
##teamcity[testCount count='4' flowId='1234']
|
||||||
|
##teamcity[testSuiteStarted name='Tests/tests/SuccessOnly' locationHint='pest_qn://tests/.tests/SuccessOnly.php' flowId='1234']
|
||||||
|
##teamcity[testCount count='4' flowId='1234']
|
||||||
##teamcity[testStarted name='it can pass with comparison' locationHint='pest_qn://tests/.tests/SuccessOnly.php::it can pass with comparison' flowId='1234']
|
##teamcity[testStarted name='it can pass with comparison' locationHint='pest_qn://tests/.tests/SuccessOnly.php::it can pass with comparison' flowId='1234']
|
||||||
|
##teamcity[testStarted name='it can pass with comparison' locationHint='pest_qn://tests/.tests/SuccessOnly.php::it can pass with comparison' flowId='1234']
|
||||||
|
##teamcity[testFinished name='it can pass with comparison' duration='100000' flowId='1234']
|
||||||
##teamcity[testFinished name='it can pass with comparison' duration='100000' flowId='1234']
|
##teamcity[testFinished name='it can pass with comparison' duration='100000' flowId='1234']
|
||||||
##teamcity[testStarted name='can also pass' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can also pass' flowId='1234']
|
##teamcity[testStarted name='can also pass' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can also pass' flowId='1234']
|
||||||
|
##teamcity[testStarted name='can also pass' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can also pass' flowId='1234']
|
||||||
|
##teamcity[testFinished name='can also pass' duration='100000' flowId='1234']
|
||||||
##teamcity[testFinished name='can also pass' duration='100000' flowId='1234']
|
##teamcity[testFinished name='can also pass' duration='100000' flowId='1234']
|
||||||
##teamcity[testSuiteStarted name='can pass with dataset' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can pass with dataset' flowId='1234']
|
##teamcity[testSuiteStarted name='can pass with dataset' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can pass with dataset' flowId='1234']
|
||||||
|
##teamcity[testSuiteStarted name='can pass with dataset' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can pass with dataset' flowId='1234']
|
||||||
|
##teamcity[testStarted name='can pass with dataset with data set "(true)"' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can pass with dataset with data set "(true)"' flowId='1234']
|
||||||
##teamcity[testStarted name='can pass with dataset with data set "(true)"' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can pass with dataset with data set "(true)"' flowId='1234']
|
##teamcity[testStarted name='can pass with dataset with data set "(true)"' locationHint='pest_qn://tests/.tests/SuccessOnly.php::can pass with dataset with data set "(true)"' flowId='1234']
|
||||||
##teamcity[testFinished name='can pass with dataset with data set "(true)"' duration='100000' flowId='1234']
|
##teamcity[testFinished name='can pass with dataset with data set "(true)"' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='can pass with dataset with data set "(true)"' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testSuiteFinished name='can pass with dataset' flowId='1234']
|
||||||
##teamcity[testSuiteFinished name='can pass with dataset' flowId='1234']
|
##teamcity[testSuiteFinished name='can pass with dataset' flowId='1234']
|
||||||
##teamcity[testSuiteStarted name='`block` → can pass with dataset in describe block' locationHint='pest_qn://tests/.tests/SuccessOnly.php::`block` → can pass with dataset in describe block' flowId='1234']
|
##teamcity[testSuiteStarted name='`block` → can pass with dataset in describe block' locationHint='pest_qn://tests/.tests/SuccessOnly.php::`block` → can pass with dataset in describe block' flowId='1234']
|
||||||
|
##teamcity[testSuiteStarted name='`block` → can pass with dataset in describe block' locationHint='pest_qn://tests/.tests/SuccessOnly.php::`block` → can pass with dataset in describe block' flowId='1234']
|
||||||
|
##teamcity[testStarted name='`block` → can pass with dataset in describe block with data set "(1)"' locationHint='pest_qn://tests/.tests/SuccessOnly.php::`block` → can pass with dataset in describe block with data set "(1)"' flowId='1234']
|
||||||
##teamcity[testStarted name='`block` → can pass with dataset in describe block with data set "(1)"' locationHint='pest_qn://tests/.tests/SuccessOnly.php::`block` → can pass with dataset in describe block with data set "(1)"' flowId='1234']
|
##teamcity[testStarted name='`block` → can pass with dataset in describe block with data set "(1)"' locationHint='pest_qn://tests/.tests/SuccessOnly.php::`block` → can pass with dataset in describe block with data set "(1)"' flowId='1234']
|
||||||
##teamcity[testFinished name='`block` → can pass with dataset in describe block with data set "(1)"' duration='100000' flowId='1234']
|
##teamcity[testFinished name='`block` → can pass with dataset in describe block with data set "(1)"' duration='100000' flowId='1234']
|
||||||
|
##teamcity[testFinished name='`block` → can pass with dataset in describe block with data set "(1)"' duration='100000' flowId='1234']
|
||||||
##teamcity[testSuiteFinished name='`block` → can pass with dataset in describe block' flowId='1234']
|
##teamcity[testSuiteFinished name='`block` → can pass with dataset in describe block' flowId='1234']
|
||||||
|
##teamcity[testSuiteFinished name='`block` → can pass with dataset in describe block' flowId='1234']
|
||||||
|
##teamcity[testSuiteFinished name='Tests/tests/SuccessOnly' flowId='1234']
|
||||||
##teamcity[testSuiteFinished name='Tests/tests/SuccessOnly' flowId='1234']
|
##teamcity[testSuiteFinished name='Tests/tests/SuccessOnly' flowId='1234']
|
||||||
|
|
||||||
[90mTests:[39m [32;1m4 passed[39;22m[90m (4 assertions)[39m
|
[90mTests:[39m [32;1m4 passed[39;22m[90m (4 assertions)[39m
|
||||||
[90mDuration:[39m [39m1.00s[39m
|
[90mDuration:[39m [39m1.00s[39m
|
||||||
|
|
||||||
|
|
||||||
|
[90mTests:[39m [32;1m4 passed[39;22m[90m (4 assertions)[39m
|
||||||
|
[90mDuration:[39m [39m1.00s[39m
|
||||||
|
|
||||||
|
|||||||
@ -1,10 +1,9 @@
|
|||||||
|
|
||||||
PASS Tests\Arch
|
PASS Tests\Arch
|
||||||
✓ preset → php → ignoring ['Pest\Expectation', 'debug_backtrace', 'var_export', …]
|
✓ preset → php → ignoring ['Pest\Expectation', 'debug_backtrace', 'var_export', …]
|
||||||
✓ preset → strict → ignoring ['usleep']
|
✓ preset → strict → ignoring ['Pest\Plugins\Tia\BaselineSync', 'usleep']
|
||||||
✓ preset → security → ignoring ['eval', 'str_shuffle', 'exec', …]
|
✓ preset → security → ignoring ['eval', 'str_shuffle', 'exec', …]
|
||||||
✓ globals
|
✓ globals
|
||||||
✓ contracts
|
|
||||||
|
|
||||||
PASS Tests\Environments\Windows
|
PASS Tests\Environments\Windows
|
||||||
✓ global functions are loaded
|
✓ global functions are loaded
|
||||||
@ -1037,8 +1036,6 @@
|
|||||||
✓ pass with toArray
|
✓ pass with toArray
|
||||||
✓ pass with array
|
✓ pass with array
|
||||||
✓ pass with toSnapshot
|
✓ pass with toSnapshot
|
||||||
✓ failures
|
|
||||||
✓ failures with custom message
|
|
||||||
✓ not failures
|
✓ not failures
|
||||||
✓ multiple snapshot expectations
|
✓ multiple snapshot expectations
|
||||||
✓ multiple snapshot expectations with datasets with (1)
|
✓ multiple snapshot expectations with datasets with (1)
|
||||||
@ -1718,6 +1715,43 @@
|
|||||||
PASS Tests\Unit\Plugins\Retry
|
PASS Tests\Unit\Plugins\Retry
|
||||||
✓ it orders by defects and stop on defects if when --retry is used
|
✓ it orders by defects and stop on defects if when --retry is used
|
||||||
|
|
||||||
|
PASS Tests\Unit\Plugins\Tia\ContentHash
|
||||||
|
✓ of() → it returns false when file does not exist
|
||||||
|
✓ of() → it hashes an existing file
|
||||||
|
✓ PHP files → it produces the same hash regardless of whitespace differences
|
||||||
|
✓ PHP files → it ignores single-line comments
|
||||||
|
✓ PHP files → it ignores hash-style comments
|
||||||
|
✓ PHP files → it ignores multi-line comments
|
||||||
|
✓ PHP files → it ignores doc comments
|
||||||
|
✓ PHP files → it detects code changes
|
||||||
|
✓ PHP files → it preserves whitespace inside string literals
|
||||||
|
✓ PHP files → it treats variable renames as a change
|
||||||
|
✓ PHP files → it falls back to a raw hash for unparseable PHP
|
||||||
|
✓ PHP files → it is case-insensitive on the file extension
|
||||||
|
✓ Blade files → it strips blade comments
|
||||||
|
✓ Blade files → it strips multi-line blade comments
|
||||||
|
✓ Blade files → it collapses whitespace
|
||||||
|
✓ Blade files → it detects content changes
|
||||||
|
✓ Blade files → it keeps blade directives intact
|
||||||
|
✓ Blade files → it does not use the PHP tokenizer for blade files
|
||||||
|
✓ JavaScript-like files → it strips line comments
|
||||||
|
✓ JavaScript-like files → it strips block comments on their own lines
|
||||||
|
✓ JavaScript-like files → it collapses whitespace
|
||||||
|
✓ JavaScript-like files → it detects code changes
|
||||||
|
✓ JavaScript-like files → it does not strip inline trailing comments
|
||||||
|
✓ JavaScript-like files → it applies the same rules to .ts files
|
||||||
|
✓ JavaScript-like files → it applies the same rules to .tsx files
|
||||||
|
✓ JavaScript-like files → it applies the same rules to .jsx files
|
||||||
|
✓ JavaScript-like files → it applies the same rules to .vue files
|
||||||
|
✓ JavaScript-like files → it applies the same rules to .svelte files
|
||||||
|
✓ JavaScript-like files → it applies the same rules to .mjs, .cjs, and .mts files
|
||||||
|
✓ unknown extensions → it hashes the raw content for unknown extensions
|
||||||
|
✓ unknown extensions → it does not normalise whitespace for unknown extensions
|
||||||
|
✓ unknown extensions → it does not strip comments for unknown extensions
|
||||||
|
✓ unknown extensions → it hashes files with no extension as raw content
|
||||||
|
✓ output format → it returns a 32-character hex xxh128 hash
|
||||||
|
✓ output format → it returns a stable hash for empty content
|
||||||
|
|
||||||
PASS Tests\Unit\Preset
|
PASS Tests\Unit\Preset
|
||||||
✓ preset invalid name
|
✓ preset invalid name
|
||||||
✓ preset → myFramework
|
✓ preset → myFramework
|
||||||
@ -1903,4 +1937,4 @@
|
|||||||
✓ pass with dataset with ('my-datas-set-value')
|
✓ pass with dataset with ('my-datas-set-value')
|
||||||
✓ within describe → pass with dataset with ('my-datas-set-value')
|
✓ within describe → pass with dataset with ('my-datas-set-value')
|
||||||
|
|
||||||
Tests: 2 deprecated, 4 warnings, 5 incomplete, 2 notices, 40 todos, 35 skipped, 1296 passed (2977 assertions)
|
Tests: 2 deprecated, 4 warnings, 5 incomplete, 2 notices, 40 todos, 35 skipped, 1328 passed (3008 assertions)
|
||||||
@ -1,15 +1,20 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
use Pest\Expectation;
|
use Pest\Expectation;
|
||||||
|
use Pest\Plugins\Tia\BaselineSync;
|
||||||
|
|
||||||
arch()->preset()->php()->ignoring([
|
arch()->preset()->php()->ignoring([
|
||||||
Expectation::class,
|
Expectation::class,
|
||||||
'debug_backtrace',
|
'debug_backtrace',
|
||||||
'var_export',
|
'var_export',
|
||||||
'xdebug_info',
|
'xdebug_info',
|
||||||
|
'xdebug_start_code_coverage',
|
||||||
|
'xdebug_stop_code_coverage',
|
||||||
|
'xdebug_get_code_coverage',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
arch()->preset()->strict()->ignoring([
|
arch()->preset()->strict()->ignoring([
|
||||||
|
BaselineSync::class,
|
||||||
'usleep',
|
'usleep',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@ -17,7 +22,9 @@ arch()->preset()->security()->ignoring([
|
|||||||
'eval',
|
'eval',
|
||||||
'str_shuffle',
|
'str_shuffle',
|
||||||
'exec',
|
'exec',
|
||||||
|
'md5',
|
||||||
'unserialize',
|
'unserialize',
|
||||||
|
'uniqid',
|
||||||
'extract',
|
'extract',
|
||||||
'assert',
|
'assert',
|
||||||
]);
|
]);
|
||||||
@ -26,13 +33,3 @@ arch('globals')
|
|||||||
->expect(['dd', 'dump', 'ray', 'die', 'var_dump', 'sleep'])
|
->expect(['dd', 'dump', 'ray', 'die', 'var_dump', 'sleep'])
|
||||||
->not->toBeUsed()
|
->not->toBeUsed()
|
||||||
->ignoring(Expectation::class);
|
->ignoring(Expectation::class);
|
||||||
|
|
||||||
arch('contracts')
|
|
||||||
->expect('Pest\Contracts')
|
|
||||||
->toOnlyUse([
|
|
||||||
'NunoMaduro\Collision\Contracts',
|
|
||||||
'Pest\Factories\TestCaseMethodFactory',
|
|
||||||
'Symfony\Component\Console',
|
|
||||||
'Pest\Arch\Contracts',
|
|
||||||
'Pest\PendingCalls',
|
|
||||||
])->toBeInterfaces();
|
|
||||||
|
|||||||
@ -134,18 +134,6 @@ test('pass with `toSnapshot`', function () {
|
|||||||
expect($object)->toMatchSnapshot();
|
expect($object)->toMatchSnapshot();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('failures', function () {
|
|
||||||
TestSuite::getInstance()->snapshots->save($this->snapshotable);
|
|
||||||
|
|
||||||
expect('contain that does not match snapshot')->toMatchSnapshot();
|
|
||||||
})->throws(ExpectationFailedException::class, 'Failed asserting that two strings are identical.');
|
|
||||||
|
|
||||||
test('failures with custom message', function () {
|
|
||||||
TestSuite::getInstance()->snapshots->save($this->snapshotable);
|
|
||||||
|
|
||||||
expect('contain that does not match snapshot')->toMatchSnapshot('oh no');
|
|
||||||
})->throws(ExpectationFailedException::class, 'oh no');
|
|
||||||
|
|
||||||
test('not failures', function () {
|
test('not failures', function () {
|
||||||
TestSuite::getInstance()->snapshots->save($this->snapshotable);
|
TestSuite::getInstance()->snapshots->save($this->snapshotable);
|
||||||
|
|
||||||
|
|||||||
@ -86,5 +86,12 @@ dataset('dataset_in_pest_file', ['A', 'B']);
|
|||||||
|
|
||||||
function removeAnsiEscapeSequences(string $input): ?string
|
function removeAnsiEscapeSequences(string $input): ?string
|
||||||
{
|
{
|
||||||
return preg_replace('#\\x1b[[][^A-Za-z]*[A-Za-z]#', '', $input);
|
return preg_replace(
|
||||||
|
[
|
||||||
|
'#\\x1b[[][^A-Za-z]*[A-Za-z]#', // CSI (colors, cursor, etc.)
|
||||||
|
'#\\x1b\\]8;[^\\x1b\\x07]*(?:\\x1b\\\\|\\x07)#', // OSC 8 hyperlinks
|
||||||
|
],
|
||||||
|
'',
|
||||||
|
$input,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
261
tests/Unit/Plugins/Tia/ContentHash.php
Normal file
261
tests/Unit/Plugins/Tia/ContentHash.php
Normal file
@ -0,0 +1,261 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Pest\Plugins\Tia\ContentHash;
|
||||||
|
|
||||||
|
describe('of()', function () {
|
||||||
|
it('returns false when file does not exist', function () {
|
||||||
|
expect(ContentHash::of('/path/that/does/not/exist.php'))->toBeFalse();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('hashes an existing file', function () {
|
||||||
|
$path = tempnam(sys_get_temp_dir(), 'pest_').'.php';
|
||||||
|
file_put_contents($path, "<?php echo 'hi';");
|
||||||
|
|
||||||
|
try {
|
||||||
|
expect(ContentHash::of($path))->toBeString()->not->toBeEmpty();
|
||||||
|
} finally {
|
||||||
|
@unlink($path);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('PHP files', function () {
|
||||||
|
it('produces the same hash regardless of whitespace differences', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', "<?php \$foo = 1;\n\necho \$foo;");
|
||||||
|
$b = ContentHash::ofContent('a.php', '<?php $foo=1; echo $foo;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores single-line comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', "<?php\n// this is a comment\n\$foo = 1;");
|
||||||
|
$b = ContentHash::ofContent('a.php', "<?php\n\$foo = 1;");
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores hash-style comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', "<?php\n# hash comment\n\$foo = 1;");
|
||||||
|
$b = ContentHash::ofContent('a.php', "<?php\n\$foo = 1;");
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores multi-line comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', "<?php\n/* a multi\n line comment */\n\$foo = 1;");
|
||||||
|
$b = ContentHash::ofContent('a.php', "<?php\n\$foo = 1;");
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores doc comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', "<?php\n/**\n * @return int\n */\nfunction foo() { return 1; }");
|
||||||
|
$b = ContentHash::ofContent('a.php', "<?php\nfunction foo() { return 1; }");
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('detects code changes', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', '<?php $foo = 1;');
|
||||||
|
$b = ContentHash::ofContent('a.php', '<?php $foo = 2;');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('preserves whitespace inside string literals', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', "<?php \$foo = 'hello world';");
|
||||||
|
$b = ContentHash::ofContent('a.php', "<?php \$foo = 'helloworld';");
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('treats variable renames as a change', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', '<?php $foo = 1;');
|
||||||
|
$b = ContentHash::ofContent('a.php', '<?php $bar = 1;');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('falls back to a raw hash for unparseable PHP', function () {
|
||||||
|
$hash = ContentHash::ofContent('a.php', 'not valid php at all');
|
||||||
|
|
||||||
|
expect($hash)->toBeString()->not->toBeEmpty();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('is case-insensitive on the file extension', function () {
|
||||||
|
$a = ContentHash::ofContent('a.PHP', "<?php\n// comment\n\$foo = 1;");
|
||||||
|
$b = ContentHash::ofContent('a.php', "<?php\n\$foo = 1;");
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Blade files', function () {
|
||||||
|
it('strips blade comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.blade.php', '<div>{{-- a comment --}}Hello</div>');
|
||||||
|
$b = ContentHash::ofContent('a.blade.php', '<div>Hello</div>');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('strips multi-line blade comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.blade.php', "<div>\n{{--\n multi\n line\n--}}\nHello\n</div>");
|
||||||
|
$b = ContentHash::ofContent('a.blade.php', '<div> Hello </div>');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('collapses whitespace', function () {
|
||||||
|
$a = ContentHash::ofContent('a.blade.php', "<div>\n Hello\n World\n</div>");
|
||||||
|
$b = ContentHash::ofContent('a.blade.php', '<div> Hello World </div>');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('detects content changes', function () {
|
||||||
|
$a = ContentHash::ofContent('a.blade.php', '<div>Hello</div>');
|
||||||
|
$b = ContentHash::ofContent('a.blade.php', '<div>Goodbye</div>');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('keeps blade directives intact', function () {
|
||||||
|
$a = ContentHash::ofContent('a.blade.php', '@if($user)Hi @endif');
|
||||||
|
$b = ContentHash::ofContent('a.blade.php', '@if($user)Bye @endif');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not use the PHP tokenizer for blade files', function () {
|
||||||
|
$a = ContentHash::ofContent('a.blade.php', '<?php // not stripped ?> hello');
|
||||||
|
$b = ContentHash::ofContent('a.blade.php', '<?php ?> hello');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('JavaScript-like files', function () {
|
||||||
|
it('strips line comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.js', "// a comment\nconst foo = 1;");
|
||||||
|
$b = ContentHash::ofContent('a.js', 'const foo = 1;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('strips block comments on their own lines', function () {
|
||||||
|
$a = ContentHash::ofContent('a.js', "/* block */\nconst foo = 1;");
|
||||||
|
$b = ContentHash::ofContent('a.js', 'const foo = 1;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('collapses whitespace', function () {
|
||||||
|
$a = ContentHash::ofContent('a.js', "const foo = 1;\n\nconst bar = 2;");
|
||||||
|
$b = ContentHash::ofContent('a.js', 'const foo = 1; const bar = 2;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('detects code changes', function () {
|
||||||
|
$a = ContentHash::ofContent('a.js', 'const foo = 1;');
|
||||||
|
$b = ContentHash::ofContent('a.js', 'const foo = 2;');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not strip inline trailing comments', function () {
|
||||||
|
$a = ContentHash::ofContent('a.js', 'const foo = 1; // inline');
|
||||||
|
$b = ContentHash::ofContent('a.js', 'const foo = 1;');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies the same rules to .ts files', function () {
|
||||||
|
$a = ContentHash::ofContent('a.ts', "// comment\nconst foo: number = 1;");
|
||||||
|
$b = ContentHash::ofContent('a.ts', 'const foo: number = 1;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies the same rules to .tsx files', function () {
|
||||||
|
$a = ContentHash::ofContent('a.tsx', "// comment\nconst Foo = () => <div/>;");
|
||||||
|
$b = ContentHash::ofContent('a.tsx', 'const Foo = () => <div/>;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies the same rules to .jsx files', function () {
|
||||||
|
$a = ContentHash::ofContent('a.jsx', "// comment\nconst Foo = () => <div/>;");
|
||||||
|
$b = ContentHash::ofContent('a.jsx', 'const Foo = () => <div/>;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies the same rules to .vue files', function () {
|
||||||
|
$a = ContentHash::ofContent('a.vue', "<script>\n// comment\nexport default {}\n</script>");
|
||||||
|
$b = ContentHash::ofContent('a.vue', '<script> export default {} </script>');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies the same rules to .svelte files', function () {
|
||||||
|
$a = ContentHash::ofContent('a.svelte', "<script>\n// comment\nlet foo = 1;\n</script>");
|
||||||
|
$b = ContentHash::ofContent('a.svelte', '<script> let foo = 1; </script>');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies the same rules to .mjs, .cjs, and .mts files', function () {
|
||||||
|
foreach (['mjs', 'cjs', 'mts'] as $ext) {
|
||||||
|
$a = ContentHash::ofContent("a.$ext", "// comment\nexport const foo = 1;");
|
||||||
|
$b = ContentHash::ofContent("a.$ext", 'export const foo = 1;');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('unknown extensions', function () {
|
||||||
|
it('hashes the raw content for unknown extensions', function () {
|
||||||
|
$a = ContentHash::ofContent('a.txt', 'hello world');
|
||||||
|
$b = ContentHash::ofContent('a.txt', 'hello world');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not normalise whitespace for unknown extensions', function () {
|
||||||
|
$a = ContentHash::ofContent('a.txt', 'hello world');
|
||||||
|
$b = ContentHash::ofContent('a.txt', 'hello world');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not strip comments for unknown extensions', function () {
|
||||||
|
$a = ContentHash::ofContent('a.txt', "// not a comment here\nhello");
|
||||||
|
$b = ContentHash::ofContent('a.txt', 'hello');
|
||||||
|
|
||||||
|
expect($a)->not->toBe($b);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('hashes files with no extension as raw content', function () {
|
||||||
|
$a = ContentHash::ofContent('Makefile', "all:\n\techo hi");
|
||||||
|
$b = ContentHash::ofContent('Makefile', "all:\n\techo hi");
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('output format', function () {
|
||||||
|
it('returns a 32-character hex xxh128 hash', function () {
|
||||||
|
$hash = ContentHash::ofContent('a.php', '<?php $foo = 1;');
|
||||||
|
|
||||||
|
expect($hash)->toMatch('/^[a-f0-9]{32}$/');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns a stable hash for empty content', function () {
|
||||||
|
$a = ContentHash::ofContent('a.php', '');
|
||||||
|
$b = ContentHash::ofContent('a.php', '');
|
||||||
|
|
||||||
|
expect($a)->toBe($b);
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -16,6 +16,7 @@ $run = function () {
|
|||||||
|
|
||||||
test('parallel', function () use ($run) {
|
test('parallel', function () use ($run) {
|
||||||
$output = $run('--exclude-group=integration');
|
$output = $run('--exclude-group=integration');
|
||||||
|
$output = implode("\n", array_slice(explode("\n", $output), -10));
|
||||||
|
|
||||||
if (getenv('REBUILD_SNAPSHOTS')) {
|
if (getenv('REBUILD_SNAPSHOTS')) {
|
||||||
preg_match('/Tests:\s+(.+\(\d+ assertions\))/', $output, $matches);
|
preg_match('/Tests:\s+(.+\(\d+ assertions\))/', $output, $matches);
|
||||||
@ -23,13 +24,13 @@ test('parallel', function () use ($run) {
|
|||||||
$file = file_get_contents(__FILE__);
|
$file = file_get_contents(__FILE__);
|
||||||
$file = preg_replace(
|
$file = preg_replace(
|
||||||
'/\$expected = \'.*?\';/',
|
'/\$expected = \'.*?\';/',
|
||||||
"\$expected = '2 deprecated, 4 warnings, 5 incomplete, 3 notices, 40 todos, 27 skipped, 1280 passed (2926 assertions)';",
|
"\$expected = '2 deprecated, 4 warnings, 5 incomplete, 3 notices, 40 todos, 27 skipped, 1312 passed (2957 assertions)';",
|
||||||
$file,
|
$file,
|
||||||
);
|
);
|
||||||
file_put_contents(__FILE__, $file);
|
file_put_contents(__FILE__, $file);
|
||||||
}
|
}
|
||||||
|
|
||||||
$expected = '2 deprecated, 4 warnings, 5 incomplete, 3 notices, 40 todos, 27 skipped, 1280 passed (2926 assertions)';
|
$expected = '2 deprecated, 4 warnings, 5 incomplete, 3 notices, 40 todos, 27 skipped, 1312 passed (2957 assertions)';
|
||||||
|
|
||||||
expect($output)
|
expect($output)
|
||||||
->toContain("Tests: {$expected}")
|
->toContain("Tests: {$expected}")
|
||||||
|
|||||||
@ -21,8 +21,10 @@ test('visual snapshot of test suite on success', function () {
|
|||||||
|
|
||||||
return preg_replace([
|
return preg_replace([
|
||||||
'#\\x1b[[][^A-Za-z]*[A-Za-z]#',
|
'#\\x1b[[][^A-Za-z]*[A-Za-z]#',
|
||||||
|
'#\\x1b\\]8;[^\\x1b\\x07]*(?:\\x1b\\\\|\\x07)#',
|
||||||
'/(Tests\\\PHPUnit\\\CustomAffixes\\\InvalidTestName)([A-Za-z0-9]*)/',
|
'/(Tests\\\PHPUnit\\\CustomAffixes\\\InvalidTestName)([A-Za-z0-9]*)/',
|
||||||
], [
|
], [
|
||||||
|
'',
|
||||||
'',
|
'',
|
||||||
'$1',
|
'$1',
|
||||||
], $process->getOutput());
|
], $process->getOutput());
|
||||||
|
|||||||
Reference in New Issue
Block a user