## Summary - cut over workspace-owned analysis and library surfaces to workspace shell ownership instead of inheriting remembered environment shell context - update the affected findings pages, scope resolution, navigation helpers, and related Blade views to keep environment focus explicit instead of implicit - add and update Spec 320 artifacts plus focused regression coverage for findings navigation context, workspace hub registration, and admin surface scope behavior ## Guardrails - Filament remains on v5 with Livewire v4 compliance unchanged - provider registration remains in apps/platform/bootstrap/providers.php - no new globally searchable resources were introduced or changed - no new destructive actions were introduced or changed - no Filament assets were added or changed, so the deploy requirement for filament:assets is unchanged ## Testing - `cd apps/platform && ./vendor/bin/sail artisan test --compact tests/Feature/Findings/FindingsAssignmentHygieneReportTest.php tests/Feature/Findings/FindingsIntakeQueueNavigationContextTest.php tests/Feature/Findings/FindingsIntakeQueueTest.php tests/Feature/Findings/MyFindingsInboxNavigationContextTest.php tests/Feature/Findings/MyWorkInboxTest.php tests/Feature/Navigation/WorkspaceHubRegistryTest.php tests/Unit/Support/OperateHub/OperateHubShellResolutionTest.php tests/Unit/Tenants/AdminSurfaceScopeTest.php` - `cd apps/platform && ./vendor/bin/sail bin pint --dirty --format agent` Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #375
153 lines
4.9 KiB
PHP
153 lines
4.9 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Support\Navigation;
|
|
|
|
use App\Support\Tenants\TenantInteractionLane;
|
|
use Illuminate\Http\Request;
|
|
|
|
enum AdminSurfaceScope: string
|
|
{
|
|
case WorkspaceWideSurface = 'workspace_wide_surface';
|
|
case WorkspaceOwnedAnalysisSurface = 'workspace_owned_analysis_surface';
|
|
case WorkspaceScoped = 'workspace_scoped';
|
|
case WorkspaceChooserException = 'workspace_chooser_exception';
|
|
case EnvironmentBound = 'environment_bound';
|
|
case EnvironmentScopedEvidence = 'environment_scoped_evidence';
|
|
case OnboardingWorkflow = 'onboarding_workflow';
|
|
case CanonicalWorkspaceRecordViewer = 'canonical_workspace_record_viewer';
|
|
|
|
public static function fromRequest(?Request $request = null): self
|
|
{
|
|
if (! $request instanceof Request) {
|
|
return self::WorkspaceScoped;
|
|
}
|
|
|
|
return self::fromPath(self::effectivePath($request));
|
|
}
|
|
|
|
public static function fromPath(string $path): self
|
|
{
|
|
$normalizedPath = '/'.ltrim($path, '/');
|
|
|
|
if ($normalizedPath === '/admin/choose-workspace') {
|
|
return self::WorkspaceChooserException;
|
|
}
|
|
|
|
if (preg_match('#^/admin/workspaces/[^/]+/operations/[^/]+$#', $normalizedPath) === 1) {
|
|
return self::CanonicalWorkspaceRecordViewer;
|
|
}
|
|
|
|
if (self::isWorkspaceWideSurfacePath($normalizedPath)) {
|
|
return self::WorkspaceWideSurface;
|
|
}
|
|
|
|
if (self::isWorkspaceOwnedAnalysisSurfacePath($normalizedPath)) {
|
|
return self::WorkspaceOwnedAnalysisSurface;
|
|
}
|
|
|
|
if (
|
|
str_starts_with($normalizedPath, '/admin/evidence/')
|
|
&& ! str_starts_with($normalizedPath, '/admin/evidence/overview')
|
|
) {
|
|
return self::EnvironmentScopedEvidence;
|
|
}
|
|
|
|
if (preg_match('#^/admin/onboarding(?:/[^/]+)?$#', $normalizedPath) === 1) {
|
|
return self::OnboardingWorkflow;
|
|
}
|
|
|
|
if (preg_match('#^/admin/workspaces/[^/]+/environments/[^/]+(?:/|$)#', $normalizedPath) === 1) {
|
|
return self::EnvironmentBound;
|
|
}
|
|
|
|
return self::WorkspaceScoped;
|
|
}
|
|
|
|
public function allowsQueryEnvironmentHints(): bool
|
|
{
|
|
return match ($this) {
|
|
self::WorkspaceWideSurface, self::WorkspaceScoped, self::OnboardingWorkflow => true,
|
|
default => false,
|
|
};
|
|
}
|
|
|
|
public function allowsRememberedEnvironmentRestore(): bool
|
|
{
|
|
return match ($this) {
|
|
self::WorkspaceScoped, self::OnboardingWorkflow, self::CanonicalWorkspaceRecordViewer => true,
|
|
default => false,
|
|
};
|
|
}
|
|
|
|
public function allowsEnvironmentlessState(): bool
|
|
{
|
|
return match ($this) {
|
|
self::WorkspaceWideSurface,
|
|
self::WorkspaceOwnedAnalysisSurface,
|
|
self::WorkspaceScoped,
|
|
self::WorkspaceChooserException,
|
|
self::OnboardingWorkflow,
|
|
self::CanonicalWorkspaceRecordViewer => true,
|
|
default => false,
|
|
};
|
|
}
|
|
|
|
public function forcesEnvironmentlessShellContext(): bool
|
|
{
|
|
return match ($this) {
|
|
self::WorkspaceWideSurface,
|
|
self::WorkspaceOwnedAnalysisSurface,
|
|
self::WorkspaceChooserException,
|
|
self::CanonicalWorkspaceRecordViewer => true,
|
|
default => false,
|
|
};
|
|
}
|
|
|
|
public function requiresExplicitEnvironment(): bool
|
|
{
|
|
return match ($this) {
|
|
self::EnvironmentBound, self::EnvironmentScopedEvidence => true,
|
|
default => false,
|
|
};
|
|
}
|
|
|
|
public function lane(): TenantInteractionLane
|
|
{
|
|
return TenantInteractionLane::fromSurfaceScope($this);
|
|
}
|
|
|
|
private static function isWorkspaceWideSurfacePath(string $normalizedPath): bool
|
|
{
|
|
return WorkspaceHubRegistry::isWorkspaceHubPath($normalizedPath);
|
|
}
|
|
|
|
private static function isWorkspaceOwnedAnalysisSurfacePath(string $normalizedPath): bool
|
|
{
|
|
return preg_match('#^/admin/(baseline-profiles|baseline-snapshots)(?:/.*)?$#', $normalizedPath) === 1
|
|
|| preg_match('#^/admin/findings/(?:my-work|intake|hygiene)/?$#', $normalizedPath) === 1
|
|
|| preg_match('#^/admin/cross-environment-compare/?$#', $normalizedPath) === 1;
|
|
}
|
|
|
|
private static function effectivePath(Request $request): string
|
|
{
|
|
$path = '/'.ltrim((string) $request->path(), '/');
|
|
|
|
if (! self::isLivewireRequestPath($path) && ! $request->headers->has('x-livewire')) {
|
|
return $path;
|
|
}
|
|
|
|
$refererPath = parse_url((string) $request->headers->get('referer', ''), PHP_URL_PATH);
|
|
|
|
return is_string($refererPath) && $refererPath !== ''
|
|
? '/'.ltrim($refererPath, '/')
|
|
: $path;
|
|
}
|
|
|
|
private static function isLivewireRequestPath(string $path): bool
|
|
{
|
|
return preg_match('#^/(?:livewire(?:-[^/]+)?/update|livewire-unit-test-endpoint)(?:/|$)#', $path) === 1;
|
|
}
|
|
}
|