## Summary - introduce a shared tenant-owned query and record-resolution canon for first-slice Filament resources - harden direct views, row actions, bulk actions, relation managers, and workspace-admin canonical viewers against wrong-tenant access - add registry-backed rollout metadata, search posture handling, architectural guards, and focused Pest coverage for scope parity and 404/403 semantics ## Included - Spec 150 package under `specs/150-tenant-owned-query-canon-and-wrong-tenant-guards/` - shared support classes: `TenantOwnedModelFamilies`, `TenantOwnedQueryScope`, `TenantOwnedRecordResolver` - shared Filament concern: `InteractsWithTenantOwnedRecords` - resource/page/policy hardening across findings, policies, policy versions, backup schedules, backup sets, restore runs, inventory items, and Entra groups - additional regression coverage for canonical tenant state, wrong-tenant record resolution, relation-manager congruence, and action-surface guardrails ## Validation - `vendor/bin/sail artisan test --compact` passed - full suite result: `2733 passed, 8 skipped` - formatting applied with `vendor/bin/sail bin pint --dirty --format agent` ## Notes - Livewire v4.0+ compliant via existing Filament v5 stack - provider registration remains in `bootstrap/providers.php` - globally searchable first-slice posture: Entra groups scoped; policies and policy versions explicitly disabled - destructive actions continue to use confirmation and policy authorization - no new Filament assets added; existing deployment flow remains unchanged, including `php artisan filament:assets` when registered assets are used Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #180
68 lines
1.9 KiB
PHP
68 lines
1.9 KiB
PHP
<?php
|
|
|
|
namespace App\Filament\Resources\BackupScheduleResource\Pages;
|
|
|
|
use App\Filament\Resources\BackupScheduleResource;
|
|
use App\Support\Filament\CanonicalAdminTenantFilterState;
|
|
use Filament\Resources\Pages\ListRecords;
|
|
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
|
|
|
class ListBackupSchedules extends ListRecords
|
|
{
|
|
protected static string $resource = BackupScheduleResource::class;
|
|
|
|
/**
|
|
* @param array<string, mixed> $arguments
|
|
* @param array<string, mixed> $context
|
|
*/
|
|
public function mountAction(string $name, array $arguments = [], array $context = []): mixed
|
|
{
|
|
if (($context['table'] ?? false) === true && filled($context['recordKey'] ?? null) && in_array($name, ['archive', 'restore', 'forceDelete'], true)) {
|
|
try {
|
|
BackupScheduleResource::resolveScopedRecordOrFail($context['recordKey']);
|
|
} catch (ModelNotFoundException) {
|
|
abort(404);
|
|
}
|
|
}
|
|
|
|
return parent::mountAction($name, $arguments, $context);
|
|
}
|
|
|
|
public function mount(): void
|
|
{
|
|
$this->syncCanonicalAdminTenantFilterState();
|
|
|
|
parent::mount();
|
|
}
|
|
|
|
protected function getHeaderActions(): array
|
|
{
|
|
return [
|
|
BackupScheduleResource::makeCreateAction()
|
|
->visible(fn (): bool => $this->tableHasRecords()),
|
|
];
|
|
}
|
|
|
|
protected function getTableEmptyStateActions(): array
|
|
{
|
|
return [
|
|
BackupScheduleResource::makeCreateAction(),
|
|
];
|
|
}
|
|
|
|
private function tableHasRecords(): bool
|
|
{
|
|
return $this->getTableRecords()->count() > 0;
|
|
}
|
|
|
|
private function syncCanonicalAdminTenantFilterState(): void
|
|
{
|
|
app(CanonicalAdminTenantFilterState::class)->sync(
|
|
$this->getTableFiltersSessionKey(),
|
|
tenantSensitiveFilters: [],
|
|
request: request(),
|
|
tenantFilterName: null,
|
|
);
|
|
}
|
|
}
|