## Summary This PR implements Spec 206 end to end and establishes the first checked-in test suite governance foundation for the platform app. Key changes: - add manifest-backed test lanes for fast-feedback, confidence, browser, heavy-governance, profiling, and junit - add budget and report helpers plus app-local artifact generation under `apps/platform/storage/logs/test-lanes` - add repo-root Sail-friendly lane/report wrappers - switch the default contributor test path to the fast-feedback lane - introduce explicit fixture profiles and cheaper defaults for shared tenant/provider test setup - add minimal/heavy factory states for tenant and provider connection setup - migrate the first high-usage and provider-sensitive tests to explicit fixture profiles - document budgets, taxonomy rules, DB reset guidance, and the full Spec 206 plan/contracts/tasks set ## Validation Executed during implementation: - focused Spec 206 guard/support/factory validation pack: 31 passed - provider-sensitive regression pack: 29 passed - first high-usage caller migration pack: 120 passed - lane routing and wrapper validation succeeded - pint completed successfully Measured lane baselines captured in docs: - fast-feedback: 176.74s - confidence: 394.38s - heavy-governance: 83.66s - browser: 128.87s - junit: 380.14s - profiling: 2701.51s - full-suite baseline anchor: 2624.60s ## Notes - Livewire v4 / Filament v5 runtime behavior is unchanged by this PR. - No new runtime routes, product UI flows, or database migrations are introduced. - Panel provider registration remains unchanged in `bootstrap/providers.php`. Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #239
116 lines
4.3 KiB
PHP
116 lines
4.3 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Tests\Support;
|
|
|
|
use InvalidArgumentException;
|
|
|
|
final class TestLaneBudget
|
|
{
|
|
public function __construct(
|
|
public readonly int $thresholdSeconds,
|
|
public readonly string $baselineSource,
|
|
public readonly string $enforcement,
|
|
public readonly string $lifecycleState,
|
|
public readonly ?int $baselineDeltaTargetPercent = null,
|
|
public readonly ?string $notes = null,
|
|
public readonly ?string $reviewCadence = null,
|
|
) {
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed> $budget
|
|
*/
|
|
public static function fromArray(array $budget): self
|
|
{
|
|
if (! isset($budget['thresholdSeconds'], $budget['baselineSource'], $budget['enforcement'], $budget['lifecycleState'])) {
|
|
throw new InvalidArgumentException('Budget declarations must define thresholdSeconds, baselineSource, enforcement, and lifecycleState.');
|
|
}
|
|
|
|
return new self(
|
|
thresholdSeconds: (int) $budget['thresholdSeconds'],
|
|
baselineSource: (string) $budget['baselineSource'],
|
|
enforcement: (string) $budget['enforcement'],
|
|
lifecycleState: (string) $budget['lifecycleState'],
|
|
baselineDeltaTargetPercent: isset($budget['baselineDeltaTargetPercent']) ? (int) $budget['baselineDeltaTargetPercent'] : null,
|
|
notes: isset($budget['notes']) ? (string) $budget['notes'] : null,
|
|
reviewCadence: isset($budget['reviewCadence']) ? (string) $budget['reviewCadence'] : null,
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @return array<string, int|float|string>
|
|
*/
|
|
public function evaluate(float $measuredSeconds): array
|
|
{
|
|
$budgetStatus = 'within-budget';
|
|
|
|
if ($measuredSeconds > $this->thresholdSeconds) {
|
|
$budgetStatus = $this->enforcement === 'warn' ? 'warning' : 'over-budget';
|
|
}
|
|
|
|
return array_filter([
|
|
'thresholdSeconds' => $this->thresholdSeconds,
|
|
'baselineSource' => $this->baselineSource,
|
|
'enforcement' => $this->enforcement,
|
|
'lifecycleState' => $this->lifecycleState,
|
|
'baselineDeltaTargetPercent' => $this->baselineDeltaTargetPercent,
|
|
'measuredSeconds' => round($measuredSeconds, 6),
|
|
'budgetStatus' => $budgetStatus,
|
|
], static fn (mixed $value): bool => $value !== null);
|
|
}
|
|
|
|
/**
|
|
* @param list<array<string, mixed>> $familyBudgets
|
|
* @param array<string, float|int> $durationsByFile
|
|
* @return list<array<string, int|float|string|array<int, string>>>
|
|
*/
|
|
public static function evaluateFamilyBudgets(array $familyBudgets, array $durationsByFile): array
|
|
{
|
|
$evaluations = [];
|
|
|
|
foreach ($familyBudgets as $familyBudget) {
|
|
$matchedSelectors = [];
|
|
$measuredSeconds = 0.0;
|
|
$selectorType = (string) ($familyBudget['selectorType'] ?? 'path');
|
|
$selectors = array_values(array_filter(
|
|
$familyBudget['selectors'] ?? [],
|
|
static fn (mixed $selector): bool => is_string($selector) && $selector !== '',
|
|
));
|
|
|
|
foreach ($durationsByFile as $filePath => $duration) {
|
|
foreach ($selectors as $selector) {
|
|
$matches = match ($selectorType) {
|
|
'file' => $filePath === $selector,
|
|
default => str_starts_with($filePath, rtrim($selector, '/')),
|
|
};
|
|
|
|
if (! $matches) {
|
|
continue;
|
|
}
|
|
|
|
$matchedSelectors[] = $selector;
|
|
$measuredSeconds += (float) $duration;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
$budget = self::fromArray([
|
|
'thresholdSeconds' => (int) $familyBudget['thresholdSeconds'],
|
|
'baselineSource' => (string) $familyBudget['baselineSource'],
|
|
'enforcement' => (string) $familyBudget['enforcement'],
|
|
'lifecycleState' => (string) $familyBudget['lifecycleState'],
|
|
]);
|
|
|
|
$evaluations[] = array_merge([
|
|
'familyId' => (string) $familyBudget['familyId'],
|
|
], $budget->evaluate($measuredSeconds), [
|
|
'matchedSelectors' => array_values(array_unique($matchedSelectors)),
|
|
]);
|
|
}
|
|
|
|
return $evaluations;
|
|
}
|
|
} |