TenantAtlas/apps/platform/app/Support/Baselines/Compare/CompareStrategySelection.php
ahmido d644265d30 Spec 203: extract baseline compare strategy (#233)
## Summary
- extract baseline compare orchestration behind an explicit strategy contract and registry
- preserve the current Intune compare path through a dedicated `IntuneCompareStrategy`
- harden compare launch and review surfaces for mixed, unsupported, incomplete, and strategy-failure truth
- add Spec 203 artifacts, focused regression coverage, and future-domain strategy proof tests

## Testing
- `cd apps/platform && ./vendor/bin/sail artisan test --compact tests/Unit/Baselines/CompareStrategyRegistryTest.php tests/Unit/Baselines/CompareSubjectResultContractTest.php tests/Feature/Baselines/BaselineCompareStrategySelectionTest.php tests/Feature/Baselines/BaselineComparePreconditionsTest.php tests/Feature/Baselines/BaselineCompareExecutionGuardTest.php tests/Feature/Baselines/BaselineCompareMatrixCompareAllActionTest.php tests/Feature/Filament/BaselineProfileCompareStartSurfaceTest.php tests/Feature/Filament/BaselineCompareLandingStartSurfaceTest.php tests/Feature/Filament/BaselineCompareLandingWhyNoFindingsTest.php tests/Feature/Filament/BaselineCompareMatrixPageTest.php tests/Feature/Filament/OperationRunBaselineTruthSurfaceTest.php`
- `cd apps/platform && ./vendor/bin/sail bin pint --dirty --format agent`

## Notes
- no new Filament panel/provider registration changes
- no global-search resource changes
- no new asset registration or deployment step changes

Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de>
Reviewed-on: #233
2026-04-13 21:17:04 +00:00

129 lines
4.3 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Support\Baselines\Compare;
use InvalidArgumentException;
final class CompareStrategySelection
{
/**
* @param list<array<string, mixed>> $matchedScopeEntries
* @param list<array<string, mixed>> $rejectedScopeEntries
* @param array<string, mixed> $diagnostics
*/
public function __construct(
public readonly StrategySelectionState $selectionState,
public readonly ?CompareStrategyKey $strategyKey,
public readonly array $matchedScopeEntries,
public readonly array $rejectedScopeEntries,
public readonly string $operatorReason,
public readonly array $diagnostics = [],
) {
if ($this->selectionState === StrategySelectionState::Supported && ! $this->strategyKey instanceof CompareStrategyKey) {
throw new InvalidArgumentException('Supported compare strategy selections require a strategy key.');
}
if (trim($this->operatorReason) === '') {
throw new InvalidArgumentException('Compare strategy selections require an operator-safe reason.');
}
}
/**
* @param list<array<string, mixed>> $matchedScopeEntries
* @param array<string, mixed> $diagnostics
*/
public static function supported(
CompareStrategyKey|string $strategyKey,
array $matchedScopeEntries,
array $diagnostics = [],
string $operatorReason = 'Compare strategy resolved successfully.',
): self {
return new self(
selectionState: StrategySelectionState::Supported,
strategyKey: CompareStrategyKey::from($strategyKey),
matchedScopeEntries: $matchedScopeEntries,
rejectedScopeEntries: [],
operatorReason: $operatorReason,
diagnostics: $diagnostics,
);
}
/**
* @param list<array<string, mixed>> $matchedScopeEntries
* @param list<array<string, mixed>> $rejectedScopeEntries
* @param array<string, mixed> $diagnostics
*/
public static function unsupported(
array $matchedScopeEntries,
array $rejectedScopeEntries,
array $diagnostics = [],
string $operatorReason = 'No compare strategy supports the selected governed subjects.',
): self {
return new self(
selectionState: StrategySelectionState::Unsupported,
strategyKey: null,
matchedScopeEntries: $matchedScopeEntries,
rejectedScopeEntries: $rejectedScopeEntries,
operatorReason: $operatorReason,
diagnostics: $diagnostics,
);
}
/**
* @param list<array<string, mixed>> $matchedScopeEntries
* @param array<string, mixed> $diagnostics
*/
public static function mixed(
array $matchedScopeEntries,
array $diagnostics = [],
string $operatorReason = 'The selected governed subjects span multiple compare strategy families.',
): self {
return new self(
selectionState: StrategySelectionState::Mixed,
strategyKey: null,
matchedScopeEntries: $matchedScopeEntries,
rejectedScopeEntries: [],
operatorReason: $operatorReason,
diagnostics: $diagnostics,
);
}
public function isSupported(): bool
{
return $this->selectionState === StrategySelectionState::Supported;
}
public function isUnsupported(): bool
{
return $this->selectionState === StrategySelectionState::Unsupported;
}
public function isMixed(): bool
{
return $this->selectionState === StrategySelectionState::Mixed;
}
/**
* @return array{
* selection_state: string,
* strategy_key: ?string,
* matched_scope_entries: list<array<string, mixed>>,
* rejected_scope_entries: list<array<string, mixed>>,
* operator_reason: string,
* diagnostics: array<string, mixed>
* }
*/
public function toArray(): array
{
return [
'selection_state' => $this->selectionState->value,
'strategy_key' => $this->strategyKey?->value,
'matched_scope_entries' => $this->matchedScopeEntries,
'rejected_scope_entries' => $this->rejectedScopeEntries,
'operator_reason' => $this->operatorReason,
'diagnostics' => $this->diagnostics,
];
}
}