## Summary - add Spec 185 workspace recovery posture visibility artifacts under `specs/185-workspace-recovery-posture-visibility` - promote tenant backup health and recovery evidence onto the workspace overview with separate metrics, attention ordering, calmness coverage, and tenant-dashboard drill-throughs - batch visible-tenant backup/recovery derivation to keep the workspace overview query-bounded - align follow-up fixes from the authoritative suite rerun, including dashboard truth-alignment fixtures, canonical backup schedule tenant context, guard-path cleanup, smoke-fixture credential removal, and robust theme asset manifest handling ## Testing - `cd apps/platform && ./vendor/bin/sail bin pint --dirty --format agent` - `cd apps/platform && ./vendor/bin/sail artisan test --compact tests/Unit/Filament/PanelThemeAssetTest.php tests/Feature/Guards/DerivedStateConsumerAdoptionGuardTest.php` - focused regression pack for the previously failing cases passed - full suite JUnit run passed: `3401` tests, `18849` assertions, `0` failures, `0` errors, `8` skips ## Notes - no new schema or persisted workspace recovery model - no provider-registration changes; Filament/Livewire stack remains on Filament v5 and Livewire v4 - no new destructive actions or global search changes Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #216
104 lines
4.2 KiB
PHP
104 lines
4.2 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
use App\Models\Tenant;
|
|
use App\Support\Workspaces\WorkspaceOverviewBuilder;
|
|
use Carbon\CarbonImmutable;
|
|
|
|
afterEach(function (): void {
|
|
CarbonImmutable::setTestNow();
|
|
});
|
|
|
|
it('orders backup-health and recovery-evidence attention by severity and suppresses calm recovery history', function (): void {
|
|
CarbonImmutable::setTestNow(CarbonImmutable::create(2026, 4, 9, 9, 0, 0, 'UTC'));
|
|
|
|
$absentTenant = Tenant::factory()->create([
|
|
'status' => 'active',
|
|
'name' => 'Absent Backup Tenant',
|
|
]);
|
|
[$user, $absentTenant] = createUserWithTenant($absentTenant, role: 'owner', workspaceRole: 'readonly');
|
|
workspaceOverviewSeedQuietTenantTruth($absentTenant);
|
|
|
|
$staleTenant = Tenant::factory()->create([
|
|
'status' => 'active',
|
|
'workspace_id' => (int) $absentTenant->workspace_id,
|
|
'name' => 'Stale Backup Tenant',
|
|
]);
|
|
createUserWithTenant($staleTenant, $user, role: 'owner', workspaceRole: 'readonly');
|
|
workspaceOverviewSeedQuietTenantTruth($staleTenant);
|
|
workspaceOverviewSeedHealthyBackup($staleTenant, [
|
|
'name' => 'Stale backup',
|
|
'completed_at' => now()->subDays(2),
|
|
]);
|
|
|
|
$degradedTenant = Tenant::factory()->create([
|
|
'status' => 'active',
|
|
'workspace_id' => (int) $absentTenant->workspace_id,
|
|
'name' => 'Degraded Backup Tenant',
|
|
]);
|
|
createUserWithTenant($degradedTenant, $user, role: 'owner', workspaceRole: 'readonly');
|
|
workspaceOverviewSeedQuietTenantTruth($degradedTenant);
|
|
workspaceOverviewSeedHealthyBackup($degradedTenant, [
|
|
'name' => 'Degraded backup',
|
|
'completed_at' => now()->subMinutes(30),
|
|
], [
|
|
'payload' => [],
|
|
'metadata' => [
|
|
'source' => 'metadata_only',
|
|
'assignments_fetch_failed' => true,
|
|
],
|
|
'assignments' => [],
|
|
]);
|
|
|
|
$weakenedTenant = Tenant::factory()->create([
|
|
'status' => 'active',
|
|
'workspace_id' => (int) $absentTenant->workspace_id,
|
|
'name' => 'Weakened Recovery Tenant',
|
|
]);
|
|
createUserWithTenant($weakenedTenant, $user, role: 'owner', workspaceRole: 'readonly');
|
|
workspaceOverviewSeedQuietTenantTruth($weakenedTenant);
|
|
$weakenedBackup = workspaceOverviewSeedHealthyBackup($weakenedTenant, [
|
|
'name' => 'Weakened backup',
|
|
'completed_at' => now()->subMinutes(25),
|
|
]);
|
|
workspaceOverviewSeedRestoreHistory($weakenedTenant, $weakenedBackup, 'follow_up');
|
|
|
|
$unvalidatedTenant = Tenant::factory()->create([
|
|
'status' => 'active',
|
|
'workspace_id' => (int) $absentTenant->workspace_id,
|
|
'name' => 'Unvalidated Recovery Tenant',
|
|
]);
|
|
createUserWithTenant($unvalidatedTenant, $user, role: 'owner', workspaceRole: 'readonly');
|
|
workspaceOverviewSeedQuietTenantTruth($unvalidatedTenant);
|
|
workspaceOverviewSeedHealthyBackup($unvalidatedTenant, [
|
|
'name' => 'Unvalidated backup',
|
|
'completed_at' => now()->subMinutes(20),
|
|
]);
|
|
|
|
$calmTenant = Tenant::factory()->create([
|
|
'status' => 'active',
|
|
'workspace_id' => (int) $absentTenant->workspace_id,
|
|
'name' => 'Calm Tenant',
|
|
]);
|
|
createUserWithTenant($calmTenant, $user, role: 'owner', workspaceRole: 'readonly');
|
|
workspaceOverviewSeedQuietTenantTruth($calmTenant);
|
|
$calmBackup = workspaceOverviewSeedHealthyBackup($calmTenant, [
|
|
'name' => 'Calm backup',
|
|
'completed_at' => now()->subMinutes(15),
|
|
]);
|
|
workspaceOverviewSeedRestoreHistory($calmTenant, $calmBackup, 'completed');
|
|
|
|
$workspace = $absentTenant->workspace()->firstOrFail();
|
|
$overview = app(WorkspaceOverviewBuilder::class)->build($workspace, $user);
|
|
$items = collect($overview['attention_items']);
|
|
|
|
expect($items)->toHaveCount(5)
|
|
->and($items->where('family', 'backup_health')->pluck('reason_context.state')->values()->all())
|
|
->toBe(['absent', 'stale', 'degraded'])
|
|
->and($items->where('family', 'recovery_evidence')->pluck('reason_context.state')->values()->all())
|
|
->toBe(['weakened', 'unvalidated'])
|
|
->and($items->pluck('tenant_label')->all())->not->toContain((string) $calmTenant->name)
|
|
->and($items->pluck('reason_context.reason')->filter()->all())->not->toContain('no_recent_issues_visible');
|
|
});
|