create(); $workspace = Workspace::factory()->create(['name' => 'Hidden Workspace']); WorkspaceMembership::factory()->create([ 'workspace_id' => (int) Workspace::factory()->create()->getKey(), 'user_id' => (int) $user->getKey(), 'role' => 'owner', ]); $this->actingAs($user) ->withSession([WorkspaceContext::SESSION_KEY => (int) $workspace->getKey()]) ->get('/admin') ->assertNotFound(); }); it('falls back to workspace-safe operations recovery when only workspace-level activity is actionable', function (): void { $tenant = \App\Models\Tenant::factory()->create(['status' => 'active']); [$user, $tenant] = createUserWithTenant($tenant, role: 'readonly', workspaceRole: 'readonly'); [$profile, $snapshot] = seedActiveBaselineForTenant($tenant); seedBaselineCompareRun($tenant, $profile, $snapshot, workspaceOverviewCompareCoverage()); $backupSet = workspaceOverviewSeedHealthyBackup($tenant, [ 'completed_at' => now()->subMinutes(10), ]); workspaceOverviewSeedRestoreHistory($tenant, $backupSet, 'completed'); \App\Models\OperationRun::factory()->tenantlessForWorkspace($tenant->workspace()->firstOrFail())->create([ 'status' => \App\Support\OperationRunStatus::Running->value, 'outcome' => \App\Support\OperationRunOutcome::Pending->value, ]); $workspace = $tenant->workspace()->firstOrFail(); $overview = app(\App\Support\Workspaces\WorkspaceOverviewBuilder::class)->build($workspace, $user); expect($overview['attention_items'])->toBe([]) ->and($overview['calmness']['is_calm'])->toBeFalse() ->and($overview['calmness']['next_action']['kind'])->toBe('operations_index') ->and($overview['calmness']['next_action']['url'])->toContain('tenant_scope=all') ->and($overview['calmness']['next_action']['url'])->toContain('activeTab=active'); }); it('uses switch workspace as the zero-tenant recovery action', function (): void { $user = User::factory()->create(); $workspace = Workspace::factory()->create(); WorkspaceMembership::factory()->create([ 'workspace_id' => (int) $workspace->getKey(), 'user_id' => (int) $user->getKey(), 'role' => 'readonly', ]); $overview = app(\App\Support\Workspaces\WorkspaceOverviewBuilder::class)->build($workspace, $user); expect($overview['calmness']['is_calm'])->toBeFalse() ->and($overview['calmness']['next_action']['kind'])->toBe('switch_workspace') ->and($overview['attention_empty_state']['action_label'])->toBe('Switch workspace'); }); it('keeps single-tenant backup and recovery metric drill-through available when the tenant dashboard stays membership-accessible', function (): void { CarbonImmutable::setTestNow(CarbonImmutable::create(2026, 4, 9, 9, 0, 0, 'UTC')); $backupTenant = Tenant::factory()->create(['status' => 'active']); [$user, $backupTenant] = createUserWithTenant($backupTenant, role: 'readonly', workspaceRole: 'readonly'); workspaceOverviewSeedQuietTenantTruth($backupTenant); $recoveryTenant = Tenant::factory()->create([ 'status' => 'active', 'workspace_id' => (int) $backupTenant->workspace_id, 'name' => 'Recovery Tenant', ]); createUserWithTenant($recoveryTenant, $user, role: 'readonly', workspaceRole: 'readonly'); workspaceOverviewSeedQuietTenantTruth($recoveryTenant); $recoveryBackup = workspaceOverviewSeedHealthyBackup($recoveryTenant, [ 'completed_at' => now()->subMinutes(20), ]); workspaceOverviewSeedRestoreHistory($recoveryTenant, $recoveryBackup, 'follow_up'); mock(CapabilityResolver::class, function ($mock): void { $mock->shouldReceive('primeMemberships')->once(); $mock->shouldReceive('isMember')->andReturnTrue(); $mock->shouldReceive('can') ->andReturnUsing(static function ($user, Tenant $tenant, string $capability): bool { return match ($capability) { Capabilities::TENANT_VIEW => false, default => false, }; }); }); $overview = app(\App\Support\Workspaces\WorkspaceOverviewBuilder::class) ->build($backupTenant->workspace()->firstOrFail(), $user); $metrics = collect($overview['summary_metrics'])->keyBy('key'); expect($metrics->get('backup_attention_tenants')['value'])->toBe(1) ->and($metrics->get('backup_attention_tenants')['destination']['kind'])->toBe('tenant_dashboard') ->and($metrics->get('backup_attention_tenants')['destination']['disabled'])->toBeFalse() ->and($metrics->get('backup_attention_tenants')['destination_url'])->toContain('/admin/t/') ->and($metrics->get('recovery_attention_tenants')['value'])->toBe(1) ->and($metrics->get('recovery_attention_tenants')['destination']['kind'])->toBe('tenant_dashboard') ->and($metrics->get('recovery_attention_tenants')['destination']['disabled'])->toBeFalse() ->and($metrics->get('recovery_attention_tenants')['destination_url'])->toContain('/admin/t/'); });