TenantAtlas/apps/platform/tests/Support/TestLaneBudget.php
ahmido 3c38192405 Spec 206: implement test suite governance foundation (#239)
## 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
2026-04-16 13:58:50 +00:00

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;
}
}