## Summary - amend the operator UI constitution and related SpecKit templates for the new UI/UX governance rules - add Spec 168 artifacts plus the tenant governance aggregate implementation used by the tenant dashboard, banner, and baseline compare landing surfaces - normalize Filament action surfaces around clickable-row inspection, grouped secondary actions, and explicit action-surface declarations across enrolled resources and pages - fix post-suite regressions in membership cache priming, finding workflow state refresh, tenant review derived-state invalidation, and tenant-bound backup-set related navigation ## Commit Series - `docs: amend operator UI constitution` - `spec: add tenant governance aggregate contract` - `feat: add tenant governance aggregate contract` - `refactor: normalize filament action surfaces` - `fix: resolve post-suite state regressions` ## Testing - `vendor/bin/sail artisan test --compact` - Result: `3176 passed, 8 skipped (17384 assertions)` ## Notes - Livewire v4 / Filament v5 stack remains unchanged - no provider registration changes; `bootstrap/providers.php` remains the relevant location - no new global-search resources or asset-registration changes in this branch Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #199
151 lines
5.0 KiB
PHP
151 lines
5.0 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Filament\Widgets\Dashboard;
|
|
|
|
use App\Models\Tenant;
|
|
use App\Support\Baselines\TenantGovernanceAggregate;
|
|
use App\Support\Baselines\TenantGovernanceAggregateResolver;
|
|
use App\Support\OpsUx\ActiveRuns;
|
|
use Filament\Facades\Filament;
|
|
use Filament\Widgets\Widget;
|
|
|
|
class NeedsAttention extends Widget
|
|
{
|
|
protected string $view = 'filament.widgets.dashboard.needs-attention';
|
|
|
|
/**
|
|
* @return array<string, mixed>
|
|
*/
|
|
protected function getViewData(): array
|
|
{
|
|
$tenant = Filament::getTenant();
|
|
|
|
if (! $tenant instanceof Tenant) {
|
|
return [
|
|
'pollingInterval' => null,
|
|
'items' => [],
|
|
'healthyChecks' => [],
|
|
];
|
|
}
|
|
|
|
$tenantId = (int) $tenant->getKey();
|
|
$aggregate = $this->governanceAggregate($tenant);
|
|
$compareAssessment = $aggregate->summaryAssessment;
|
|
|
|
$items = [];
|
|
|
|
$overdueOpenCount = $aggregate->overdueOpenFindingsCount;
|
|
$lapsedGovernanceCount = $aggregate->lapsedGovernanceCount;
|
|
$expiringGovernanceCount = $aggregate->expiringGovernanceCount;
|
|
$highSeverityCount = $aggregate->highSeverityActiveFindingsCount;
|
|
|
|
if ($lapsedGovernanceCount > 0) {
|
|
$items[] = [
|
|
'title' => 'Lapsed accepted-risk governance',
|
|
'body' => "{$lapsedGovernanceCount} finding(s) need governance follow-up before accepted risk is safe to rely on.",
|
|
'badge' => 'Governance',
|
|
'badgeColor' => 'danger',
|
|
];
|
|
}
|
|
|
|
if ($overdueOpenCount > 0) {
|
|
$items[] = [
|
|
'title' => 'Overdue findings',
|
|
'body' => "{$overdueOpenCount} open finding(s) are overdue and still need workflow follow-up.",
|
|
'badge' => 'Findings',
|
|
'badgeColor' => 'danger',
|
|
];
|
|
}
|
|
|
|
if ($expiringGovernanceCount > 0) {
|
|
$items[] = [
|
|
'title' => 'Expiring accepted-risk governance',
|
|
'body' => "{$expiringGovernanceCount} finding(s) will need governance review soon.",
|
|
'badge' => 'Governance',
|
|
'badgeColor' => 'warning',
|
|
];
|
|
}
|
|
|
|
if ($highSeverityCount > 0) {
|
|
$items[] = [
|
|
'title' => 'High severity active findings',
|
|
'body' => "{$highSeverityCount} active finding(s) need review.",
|
|
'badge' => 'Drift',
|
|
'badgeColor' => 'danger',
|
|
];
|
|
}
|
|
|
|
if ($compareAssessment->stateFamily !== 'positive') {
|
|
$items[] = [
|
|
'title' => 'Baseline compare posture',
|
|
'body' => $compareAssessment->headline,
|
|
'supportingMessage' => $compareAssessment->supportingMessage,
|
|
'badge' => 'Baseline',
|
|
'badgeColor' => $compareAssessment->tone,
|
|
'nextStep' => $aggregate->nextActionLabel,
|
|
];
|
|
}
|
|
|
|
$activeRuns = ActiveRuns::existForTenant($tenant)
|
|
? (int) \App\Models\OperationRun::query()->where('tenant_id', $tenantId)->active()->count()
|
|
: 0;
|
|
|
|
if ($activeRuns > 0) {
|
|
$items[] = [
|
|
'title' => 'Operations in progress',
|
|
'body' => "{$activeRuns} run(s) are active.",
|
|
'badge' => 'Operations',
|
|
'badgeColor' => 'warning',
|
|
];
|
|
}
|
|
|
|
$items = array_slice($items, 0, 5);
|
|
|
|
$healthyChecks = [];
|
|
|
|
if ($items === []) {
|
|
$healthyChecks = [
|
|
[
|
|
'title' => 'Baseline compare looks trustworthy',
|
|
'body' => $aggregate->headline,
|
|
],
|
|
[
|
|
'title' => 'No overdue findings',
|
|
'body' => 'No open findings are currently overdue for this tenant.',
|
|
],
|
|
[
|
|
'title' => 'Accepted-risk governance is healthy',
|
|
'body' => 'No accepted-risk findings currently need governance follow-up.',
|
|
],
|
|
[
|
|
'title' => 'No high severity active findings',
|
|
'body' => 'No high severity findings are currently open for this tenant.',
|
|
],
|
|
[
|
|
'title' => 'No active operations',
|
|
'body' => 'Nothing is currently running for this tenant.',
|
|
],
|
|
];
|
|
}
|
|
|
|
return [
|
|
'pollingInterval' => ActiveRuns::existForTenant($tenant) ? '10s' : null,
|
|
'items' => $items,
|
|
'healthyChecks' => $healthyChecks,
|
|
];
|
|
}
|
|
|
|
private function governanceAggregate(Tenant $tenant): TenantGovernanceAggregate
|
|
{
|
|
/** @var TenantGovernanceAggregateResolver $resolver */
|
|
$resolver = app(TenantGovernanceAggregateResolver::class);
|
|
|
|
/** @var TenantGovernanceAggregate $aggregate */
|
|
$aggregate = $resolver->forTenant($tenant);
|
|
|
|
return $aggregate;
|
|
}
|
|
}
|