TenantAtlas/tests/Unit/Support/OperatorExplanation/OperatorExplanationBuilderTest.php
ahmido 1f0cc5de56 feat: implement operator explanation layer (#191)
## Summary
- add the shared operator explanation layer with explanation families, trustworthiness semantics, count descriptors, and centralized badge mappings
- adopt explanation-first rendering across baseline compare, governance operation run detail, baseline snapshot presentation, tenant review detail, and review register rows
- extend reason translation, artifact-truth presentation, fallback ops UX messaging, and focused regression coverage for operator explanation semantics

## Testing
- vendor/bin/sail bin pint --dirty --format agent
- vendor/bin/sail artisan test --compact tests/Feature/Monitoring/OperationsTenantScopeTest.php tests/Feature/Operations/OperationRunBlockedExecutionPresentationTest.php
- vendor/bin/sail artisan test --compact

## Notes
- Livewire v4 compatible
- panel provider registration remains in bootstrap/providers.php
- no destructive Filament actions were added or changed in this PR
- no new global-search behavior was introduced in this slice

Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de>
Reviewed-on: #191
2026-03-24 11:24:33 +00:00

67 lines
3.2 KiB
PHP

<?php
declare(strict_types=1);
use App\Support\Ui\OperatorExplanation\CountDescriptor;
use App\Support\Ui\OperatorExplanation\ExplanationFamily;
use App\Support\Ui\OperatorExplanation\OperatorExplanationBuilder;
use App\Support\Ui\OperatorExplanation\TrustworthinessLevel;
use Tests\Feature\Concerns\BuildsOperatorExplanationFixtures;
uses(BuildsOperatorExplanationFixtures::class);
it('maps blocked artifact truth into an explanation-first pattern', function (): void {
$reason = $this->makeExplanationReasonEnvelope([
'internalCode' => 'review_publish_blocked',
'operatorLabel' => 'Publication blocked',
'shortExplanation' => 'A required approval or prerequisite is missing for this review.',
'trustImpact' => TrustworthinessLevel::Unusable->value,
'absencePattern' => 'blocked_prerequisite',
'nextSteps' => [\App\Support\ReasonTranslation\NextStepOption::instruction('Resolve review blockers before publication.')],
]);
$truth = $this->makeArtifactTruthEnvelope([
'executionOutcome' => 'blocked',
'artifactExistence' => 'created_but_not_usable',
'contentState' => 'missing_input',
'actionability' => 'required',
'primaryLabel' => 'Artifact not usable',
'primaryExplanation' => 'The review exists, but it is blocked from publication.',
'nextActionLabel' => 'Resolve review blockers before publication',
], $reason);
$explanation = app(OperatorExplanationBuilder::class)->fromArtifactTruthEnvelope($truth, [
new CountDescriptor('Publish blockers', 2, CountDescriptor::ROLE_RELIABILITY_SIGNAL, 'resolve before publish'),
]);
expect($explanation->family)->toBe(ExplanationFamily::BlockedPrerequisite)
->and($explanation->evaluationResult)->toBe('unavailable')
->and($explanation->trustworthinessLevel)->toBe(TrustworthinessLevel::Unusable)
->and($explanation->dominantCauseLabel)->toBe('Publication blocked')
->and($explanation->dominantCauseExplanation)->toContain('missing for this review')
->and($explanation->nextActionText)->toBe('Resolve review blockers before publication')
->and($explanation->countDescriptors)->toHaveCount(1);
});
it('keeps trustworthy artifact truth separate from no-action guidance', function (): void {
$truth = $this->makeArtifactTruthEnvelope([
'executionOutcome' => 'succeeded',
'artifactExistence' => 'created',
'contentState' => 'trusted',
'freshnessState' => 'current',
'actionability' => 'none',
'primaryLabel' => 'Trustworthy artifact',
'primaryExplanation' => 'The artifact is ready for the intended operator task.',
]);
$explanation = app(OperatorExplanationBuilder::class)->fromArtifactTruthEnvelope($truth, [
new CountDescriptor('Findings', 3, CountDescriptor::ROLE_EVALUATION_OUTPUT),
]);
expect($explanation->family)->toBe(ExplanationFamily::TrustworthyResult)
->and($explanation->evaluationResult)->toBe('full_result')
->and($explanation->trustworthinessLevel)->toBe(TrustworthinessLevel::Trustworthy)
->and($explanation->nextActionText)->toBe('No action needed')
->and($explanation->coverageStatement)->toContain('sufficient');
});