## Summary - Removes the legacy Tenant CRUD create page (`/admin/tenants/create`) so tenant creation is handled exclusively via the onboarding wizard. - Updates tenant selection flows and pages to prevent Livewire polling/notification-related 404s on workspace-scoped routes. - Aligns empty-state UX with enterprise patterns (avoid duplicate CTAs). ## Key changes - Tenant creation - Removed `CreateTenant` page + route from `TenantResource`. - `TenantResource::canCreate()` now returns `false` (CRUD creation disabled). - Tenants list now surfaces an **Add tenant** action that links to onboarding (`admin.onboarding`). - Onboarding wizard - Removed redundant legacy step-cards from the blade view (Wizard schema is the source of truth). - Disabled topbar on the onboarding page to avoid lazy-loaded notifications. - Choose tenant - Enterprise UI redesign + workspace context. - Uses Livewire `selectTenant()` instead of a form POST. - Disabled topbar and gated BODY_END hook to avoid background polling. - Baseline profiles - Hide header create action when table is empty to avoid duplicate CTAs. ## Tests - `vendor/bin/sail artisan test --compact --filter='Onboarding|ManagedTenantOnboarding'` - `vendor/bin/sail artisan test --compact --filter='ManagedTenantsLivewireUpdate'` - `vendor/bin/sail artisan test --compact --filter='TenantSetup|TenantResourceAuth|TenantAdminAuth|ListTenants'` - `vendor/bin/sail artisan test --compact --filter='BaselineProfile'` - `vendor/bin/sail artisan test --compact --filter='ChooseTenant|TenantMake|TenantScoping|AdminTenantScoped|AdminHomeRedirect|WorkspaceContext'` ## Notes - Filament v5 / Livewire v4 compatible. - No new assets introduced; no deploy pipeline changes required. Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #131
82 lines
2.7 KiB
PHP
82 lines
2.7 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
use App\Models\User;
|
|
use App\Models\Workspace;
|
|
use App\Models\WorkspaceMembership;
|
|
use App\Support\Workspaces\WorkspaceContext;
|
|
use Illuminate\Foundation\Testing\RefreshDatabase;
|
|
use Illuminate\Support\Facades\Http;
|
|
|
|
uses(RefreshDatabase::class);
|
|
|
|
beforeEach(function (): void {
|
|
Http::preventStrayRequests();
|
|
});
|
|
|
|
it('serves the managed-tenants page without Livewire update failures', function (): void {
|
|
$user = User::factory()->create();
|
|
|
|
$workspace = Workspace::factory()->create(['slug' => 'test-ws']);
|
|
WorkspaceMembership::factory()->create([
|
|
'workspace_id' => $workspace->getKey(),
|
|
'user_id' => $user->getKey(),
|
|
'role' => 'owner',
|
|
]);
|
|
|
|
// 1. Load the page
|
|
$response = $this->actingAs($user)
|
|
->withSession([WorkspaceContext::SESSION_KEY => (int) $workspace->getKey()])
|
|
->get('/admin/w/'.$workspace->slug.'/managed-tenants');
|
|
|
|
$response->assertSuccessful();
|
|
|
|
$html = $response->getContent();
|
|
|
|
// This landing page must not include Livewire-driven panel widgets that
|
|
// trigger background updates (lazy-loaded database notifications, progress poller).
|
|
expect($html)->not->toContain('Filament\\Livewire\\DatabaseNotifications');
|
|
expect($html)->not->toContain('__lazyLoad');
|
|
expect($html)->not->toContain('opsUxProgressWidgetPoller');
|
|
|
|
// 2. Extract the first Livewire component snapshot
|
|
preg_match('/wire:snapshot="([^"]+)"/', $html, $snapshotMatch);
|
|
expect($snapshotMatch)->not->toBeEmpty('No Livewire snapshot found in page HTML');
|
|
|
|
$snapshotJson = htmlspecialchars_decode($snapshotMatch[1]);
|
|
$snapshot = json_decode($snapshotJson, true);
|
|
|
|
expect($snapshot)->toBeArray();
|
|
expect($snapshot['memo']['path'] ?? null)->toBe('admin/w/test-ws/managed-tenants');
|
|
|
|
// 3. POST a Livewire update request
|
|
$updatePayload = [
|
|
'components' => [[
|
|
'snapshot' => $snapshotJson,
|
|
'updates' => new \stdClass,
|
|
'calls' => [],
|
|
]],
|
|
];
|
|
|
|
// Get the Livewire update URI path (includes hash prefix)
|
|
$routes = app('router')->getRoutes();
|
|
$updateRoute = null;
|
|
foreach ($routes as $route) {
|
|
if (str_contains($route->getName() ?? '', 'livewire.update')) {
|
|
$updateRoute = $route;
|
|
break;
|
|
}
|
|
}
|
|
expect($updateRoute)->not->toBeNull('Livewire update route must exist');
|
|
|
|
$updateResponse = $this->actingAs($user)
|
|
->withSession([WorkspaceContext::SESSION_KEY => (int) $workspace->getKey()])
|
|
->withHeaders([
|
|
'X-Livewire' => 'true',
|
|
])
|
|
->postJson('/'.$updateRoute->uri(), $updatePayload);
|
|
|
|
$updateResponse->assertSuccessful();
|
|
});
|