Implements specs 070–072 (workspace foundation, workspace-scoped tenant selection, managed-tenants workspace enforcement).
Highlights
- Adds Workspace + WorkspaceMembership models/migrations + middleware to persist/enforce current workspace context.
- Scopes tenant selection to the current workspace.
- Makes legacy `/admin/managed-tenants*` routes redirect into workspace-scoped URLs.
- Enforces tenant routes under `/admin/t/{tenant}` to 404 when workspace context is missing or mismatched.
- Fixes Filament page Blade wrappers so header actions render on choose-workspace / choose-tenant / no-access pages.
Verification
- Pint: `vendor/bin/sail bin pint --dirty`
- Tests: `vendor/bin/sail artisan test --compact tests/Feature/Guards/NoAdHocFilamentAuthPatternsTest.php tests/Feature/Workspaces tests/Feature/Filament/ChooseTenantIsWorkspaceScopedTest.php tests/Feature/Filament/ChooseTenantRequiresWorkspaceTest.php tests/Feature/Filament/TenantSwitcherUrlResolvesTenantTest.php tests/Feature/ManagedTenants tests/Feature/AdminNewRedirectTest.php`
Notes
- Filament v5 / Livewire v4 compatible.
- Panel provider registration stays in `bootstrap/providers.php` (Laravel 11+ rule).
- No new heavy frontend assets added.
Co-authored-by: Ahmed Darrazi <ahmeddarrazi@MacBookPro.fritz.box>
Reviewed-on: #85
41 lines
1.1 KiB
PHP
41 lines
1.1 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Services\Audit;
|
|
|
|
use App\Models\AuditLog;
|
|
use App\Models\User;
|
|
use App\Models\Workspace;
|
|
use Carbon\CarbonImmutable;
|
|
|
|
class WorkspaceAuditLogger
|
|
{
|
|
public function log(
|
|
Workspace $workspace,
|
|
string $action,
|
|
array $context = [],
|
|
?User $actor = null,
|
|
string $status = 'success',
|
|
?string $resourceType = null,
|
|
?string $resourceId = null,
|
|
): AuditLog {
|
|
$metadata = $context['metadata'] ?? [];
|
|
unset($context['metadata']);
|
|
|
|
return AuditLog::create([
|
|
'tenant_id' => null,
|
|
'workspace_id' => (int) $workspace->getKey(),
|
|
'actor_id' => $actor?->getKey(),
|
|
'actor_email' => $actor?->email,
|
|
'actor_name' => $actor?->name,
|
|
'action' => $action,
|
|
'resource_type' => $resourceType,
|
|
'resource_id' => $resourceId,
|
|
'status' => $status,
|
|
'metadata' => $metadata + $context,
|
|
'recorded_at' => CarbonImmutable::now(),
|
|
]);
|
|
}
|
|
}
|