## Summary - add the shared resolved-reference foundation with registry, resolvers, presenters, and badge semantics - refactor related context, assignment evidence, and policy-version assignment rendering toward label-first reference presentation - add Spec 132 artifacts and focused Pest coverage for reference resolution, degraded states, canonical linking, and tenant-context carryover ## Verification - `vendor/bin/sail bin pint --dirty --format agent` - focused Pest verification was marked complete in the task artifact ## Notes - this PR is opened from the current session branch - `specs/132-guid-context-resolver/tasks.md` reflects in-progress completion state for the implemented tasks Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #161
85 lines
2.7 KiB
PHP
85 lines
2.7 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Support\References\Resolvers;
|
|
|
|
use App\Filament\Resources\BaselineProfileResource;
|
|
use App\Models\BaselineProfile;
|
|
use App\Models\User;
|
|
use App\Models\Workspace;
|
|
use App\Services\Auth\WorkspaceCapabilityResolver;
|
|
use App\Support\Auth\Capabilities;
|
|
use App\Support\References\ReferenceClass;
|
|
use App\Support\References\ReferenceDescriptor;
|
|
use App\Support\References\ReferenceLinkTarget;
|
|
use App\Support\References\ResolvedReference;
|
|
|
|
final class BaselineProfileReferenceResolver extends BaseReferenceResolver
|
|
{
|
|
public function __construct(
|
|
\App\Support\References\ReferenceTypeLabelCatalog $typeLabels,
|
|
private readonly WorkspaceCapabilityResolver $workspaceCapabilityResolver,
|
|
) {
|
|
parent::__construct($typeLabels);
|
|
}
|
|
|
|
public function referenceClass(): ReferenceClass
|
|
{
|
|
return ReferenceClass::BaselineProfile;
|
|
}
|
|
|
|
public function resolve(ReferenceDescriptor $descriptor): ResolvedReference
|
|
{
|
|
$profileId = $this->linkedModelId($descriptor);
|
|
$workspaceId = $descriptor->workspaceId;
|
|
|
|
if ($profileId === null || $workspaceId === null || $workspaceId <= 0) {
|
|
return $this->unresolved($descriptor);
|
|
}
|
|
|
|
$profile = BaselineProfile::query()
|
|
->whereKey($profileId)
|
|
->where('workspace_id', $workspaceId)
|
|
->first();
|
|
|
|
if (! $profile instanceof BaselineProfile) {
|
|
return $this->missing($descriptor);
|
|
}
|
|
|
|
if (! $this->canOpenWorkspaceBaselines((int) $profile->workspace_id)) {
|
|
return $this->inaccessible($descriptor);
|
|
}
|
|
|
|
return $this->resolved(
|
|
descriptor: $descriptor,
|
|
primaryLabel: (string) $profile->name,
|
|
secondaryLabel: 'Baseline profile #'.$profile->getKey(),
|
|
linkTarget: new ReferenceLinkTarget(
|
|
targetKind: ReferenceClass::BaselineProfile->value,
|
|
url: BaselineProfileResource::getUrl('view', ['record' => $profile], panel: 'admin'),
|
|
actionLabel: 'View baseline profile',
|
|
contextBadge: 'Workspace',
|
|
),
|
|
);
|
|
}
|
|
|
|
private function canOpenWorkspaceBaselines(int $workspaceId): bool
|
|
{
|
|
$user = auth()->user();
|
|
|
|
if (! $user instanceof User) {
|
|
return false;
|
|
}
|
|
|
|
$workspace = Workspace::query()->whereKey($workspaceId)->first();
|
|
|
|
if (! $workspace instanceof Workspace) {
|
|
return false;
|
|
}
|
|
|
|
return $this->workspaceCapabilityResolver->isMember($user, $workspace)
|
|
&& $this->workspaceCapabilityResolver->can($user, $workspace, Capabilities::WORKSPACE_BASELINES_VIEW);
|
|
}
|
|
}
|