TenantAtlas/app/Services/Tenants/TenantOperabilityService.php
ahmido 417df4f9aa feat: central tenant operability policy (#177)
## Summary
- centralize tenant operability into a lane-aware, actor-aware policy boundary
- align selector eligibility, administrative discoverability, remembered context, tenant-bound routes, and canonical run viewers
- add focused Pest coverage plus Spec 148 artifacts and final polish task completion

## Validation
- `vendor/bin/sail artisan test --compact tests/Unit/Tenants/TenantOperabilityServiceTest.php tests/Unit/Tenants/TenantOperabilityOutcomeTest.php tests/Feature/Workspaces/ChooseTenantPageTest.php tests/Feature/Workspaces/SelectTenantControllerTest.php tests/Feature/TenantRBAC/ArchivedTenantRouteAccessTest.php tests/Feature/TenantRBAC/TenantRouteDenyAsNotFoundTest.php tests/Feature/Operations/TenantlessOperationRunViewerTest.php tests/Feature/OpsUx/OperateHubShellTest.php tests/Feature/Rbac/TenantLifecycleActionVisibilityTest.php tests/Feature/TenantRBAC/TenantSwitcherScopeTest.php tests/Feature/Rbac/TenantResourceAuthorizationTest.php tests/Feature/Filament/ManagedTenantsLandingLifecycleTest.php tests/Feature/Filament/TenantGlobalSearchLifecycleScopeTest.php tests/Feature/Onboarding/OnboardingDraftLifecycleTest.php tests/Feature/Onboarding/OnboardingDraftAuthorizationTest.php`
- `vendor/bin/sail bin pint --dirty --format agent`
- manual browser smoke checks on `/admin/choose-tenant`, `/admin/tenants`, `/admin/onboarding`, `/admin/onboarding/{draft}`, and `/admin/operations/{run}`

## Filament / platform notes
- Livewire v4 compliance preserved
- panel provider registration unchanged in `bootstrap/providers.php`
- Tenant resource global search remains backed by existing view/edit pages and is now separated from active-only selector eligibility
- destructive actions remain action closures with confirmation and authorization enforcement
- no asset pipeline changes and no new `filament:assets` deployment requirement

Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de>
Reviewed-on: #177
2026-03-17 11:48:55 +00:00

534 lines
23 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Services\Tenants;
use App\Models\Tenant;
use App\Models\TenantOnboardingSession;
use App\Models\User;
use App\Services\Auth\CapabilityResolver;
use App\Support\Tenants\TenantInteractionLane;
use App\Support\Tenants\TenantLifecycle;
use App\Support\Tenants\TenantOperabilityContext;
use App\Support\Tenants\TenantOperabilityDecision;
use App\Support\Tenants\TenantOperabilityOutcome;
use App\Support\Tenants\TenantOperabilityQuestion;
use App\Support\Tenants\TenantOperabilityReasonCode;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Support\Collection;
class TenantOperabilityService
{
public function __construct(
private readonly CapabilityResolver $capabilityResolver,
) {}
public function decisionFor(Tenant $tenant): TenantOperabilityDecision
{
return TenantOperabilityDecision::fromOutcomes([
TenantOperabilityQuestion::SelectorEligibility->value => $this->evaluate(
TenantOperabilityContext::forTenant(
tenant: $tenant,
lane: TenantInteractionLane::StandardActiveOperating,
),
TenantOperabilityQuestion::SelectorEligibility,
),
TenantOperabilityQuestion::AdministrativeDiscoverability->value => $this->evaluate(
TenantOperabilityContext::forTenant(
tenant: $tenant,
lane: TenantInteractionLane::AdministrativeManagement,
),
TenantOperabilityQuestion::AdministrativeDiscoverability,
),
TenantOperabilityQuestion::ArchiveEligibility->value => $this->evaluate(
TenantOperabilityContext::forTenant(
tenant: $tenant,
lane: TenantInteractionLane::AdministrativeManagement,
),
TenantOperabilityQuestion::ArchiveEligibility,
),
TenantOperabilityQuestion::RestoreEligibility->value => $this->evaluate(
TenantOperabilityContext::forTenant(
tenant: $tenant,
lane: TenantInteractionLane::AdministrativeManagement,
),
TenantOperabilityQuestion::RestoreEligibility,
),
TenantOperabilityQuestion::ResumeOnboardingEligibility->value => $this->evaluate(
TenantOperabilityContext::forTenant(
tenant: $tenant,
lane: TenantInteractionLane::AdministrativeManagement,
),
TenantOperabilityQuestion::ResumeOnboardingEligibility,
),
TenantOperabilityQuestion::CanonicalLinkedRecordViewability->value => $this->evaluate(
TenantOperabilityContext::forTenant(
tenant: $tenant,
lane: TenantInteractionLane::CanonicalWorkspaceRecord,
),
TenantOperabilityQuestion::CanonicalLinkedRecordViewability,
),
]);
}
public function evaluate(TenantOperabilityContext $context, TenantOperabilityQuestion $question): TenantOperabilityOutcome
{
$lifecycle = TenantLifecycle::fromTenant($context->tenant);
if ($context->workspaceId !== null && (int) $context->tenant->workspace_id !== $context->workspaceId) {
return TenantOperabilityOutcome::deny(
question: $question,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: $question === TenantOperabilityQuestion::RememberedContextValidity
? TenantOperabilityReasonCode::RememberedContextStale
: TenantOperabilityReasonCode::WorkspaceMismatch,
discoverable: false,
);
}
if ($context->actor instanceof User && ! $this->capabilityResolver->isMember($context->actor, $context->tenant)) {
return TenantOperabilityOutcome::deny(
question: $question,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::TenantNotEntitled,
discoverable: false,
metadata: $this->metadata($context),
);
}
if ($context->requiredCapability !== null && $context->actor instanceof User && ! $this->capabilityResolver->can($context->actor, $context->tenant, $context->requiredCapability)) {
return TenantOperabilityOutcome::deny(
question: $question,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::MissingCapability,
discoverable: $question === TenantOperabilityQuestion::AdministrativeDiscoverability || $question === TenantOperabilityQuestion::TenantBoundViewability,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
return match ($question) {
TenantOperabilityQuestion::SelectorEligibility => $this->selectorEligibilityOutcome($context, $lifecycle),
TenantOperabilityQuestion::RememberedContextValidity => $this->rememberedContextOutcome($context, $lifecycle),
TenantOperabilityQuestion::TenantBoundViewability => $this->tenantBoundViewabilityOutcome($context, $lifecycle),
TenantOperabilityQuestion::CanonicalLinkedRecordViewability => $this->canonicalViewabilityOutcome($context, $lifecycle),
TenantOperabilityQuestion::ArchiveEligibility => $this->archiveEligibilityOutcome($context, $lifecycle),
TenantOperabilityQuestion::RestoreEligibility => $this->restoreEligibilityOutcome($context, $lifecycle),
TenantOperabilityQuestion::ResumeOnboardingEligibility => $this->resumeOnboardingOutcome($context, $lifecycle),
TenantOperabilityQuestion::OnboardingCompletionEligibility => $this->onboardingCompletionOutcome($context, $lifecycle),
TenantOperabilityQuestion::VerificationReadinessEligibility => $this->verificationReadinessOutcome($context, $lifecycle),
TenantOperabilityQuestion::AdministrativeDiscoverability => $this->administrativeDiscoverabilityOutcome($context, $lifecycle),
};
}
public function outcomeFor(
Tenant $tenant,
TenantOperabilityQuestion $question,
?User $actor = null,
?int $workspaceId = null,
TenantInteractionLane $lane = TenantInteractionLane::AdministrativeManagement,
?TenantOnboardingSession $onboardingDraft = null,
?string $requiredCapability = null,
?Tenant $selectedTenant = null,
?string $linkedRecordType = null,
?int $linkedRecordId = null,
): TenantOperabilityOutcome {
return $this->evaluate(
TenantOperabilityContext::forTenant(
tenant: $tenant,
actor: $actor,
workspaceId: $workspaceId,
lane: $lane,
onboardingDraft: $onboardingDraft,
requiredCapability: $requiredCapability,
selectedTenant: $selectedTenant,
linkedRecordType: $linkedRecordType,
linkedRecordId: $linkedRecordId,
),
$question,
);
}
public function lifecycleFor(Tenant $tenant): TenantLifecycle
{
return $this->decisionFor($tenant)->lifecycle;
}
public function canSelectAsContext(Tenant $tenant): bool
{
return $this->outcomeFor(
tenant: $tenant,
question: TenantOperabilityQuestion::SelectorEligibility,
lane: TenantInteractionLane::StandardActiveOperating,
)->allowed;
}
public function canViewTenantSurface(Tenant $tenant): bool
{
return $this->outcomeFor(
tenant: $tenant,
question: TenantOperabilityQuestion::AdministrativeDiscoverability,
lane: TenantInteractionLane::AdministrativeManagement,
)->allowed;
}
public function canResumeOnboarding(Tenant $tenant): bool
{
return $this->outcomeFor(
tenant: $tenant,
question: TenantOperabilityQuestion::ResumeOnboardingEligibility,
lane: TenantInteractionLane::AdministrativeManagement,
)->allowed;
}
public function canArchive(Tenant $tenant): bool
{
return $this->outcomeFor(
tenant: $tenant,
question: TenantOperabilityQuestion::ArchiveEligibility,
lane: TenantInteractionLane::AdministrativeManagement,
)->allowed;
}
public function canRestore(Tenant $tenant): bool
{
return $this->outcomeFor(
tenant: $tenant,
question: TenantOperabilityQuestion::RestoreEligibility,
lane: TenantInteractionLane::AdministrativeManagement,
)->allowed;
}
public function primaryManagementActionKey(Tenant $tenant, bool $preferOnboarding = false): ?string
{
return $this->decisionFor($tenant)->primaryManagementActionKey($preferOnboarding);
}
public function canReferenceInWorkspaceMonitoring(Tenant $tenant): bool
{
return $this->outcomeFor(
tenant: $tenant,
question: TenantOperabilityQuestion::CanonicalLinkedRecordViewability,
lane: TenantInteractionLane::CanonicalWorkspaceRecord,
)->allowed;
}
/**
* @param Collection<int, Tenant> $tenants
* @return Collection<int, Tenant>
*/
public function filterSelectable(Collection $tenants): Collection
{
return $tenants
->filter(fn (mixed $tenant): bool => $tenant instanceof Tenant && $this->canSelectAsContext($tenant))
->values();
}
public function applySelectableScope(Builder $query, ?string $table = null): Builder
{
$prefix = $table !== null && $table !== '' ? "{$table}." : '';
return $query
->whereNull("{$prefix}deleted_at")
->where("{$prefix}status", TenantLifecycle::Active->value);
}
public function applyAdministrativeDiscoverabilityScope(Builder $query, ?string $table = null): Builder
{
$prefix = $table !== null && $table !== '' ? "{$table}." : '';
return $query
->withTrashed()
->where(function (Builder $builder) use ($prefix): void {
$builder->whereIn("{$prefix}status", TenantLifecycle::values())
->orWhereNotNull("{$prefix}deleted_at");
});
}
private function selectorEligibilityOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if ($context->lane !== TenantInteractionLane::StandardActiveOperating) {
return $this->wrongLaneOutcome($context, TenantOperabilityQuestion::SelectorEligibility, $lifecycle);
}
if (! $lifecycle->isSelectableInStandardLane()) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::SelectorEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::SelectorIneligibleLifecycle,
discoverable: false,
metadata: $this->metadata($context),
);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::SelectorEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
metadata: $this->metadata($context),
);
}
private function rememberedContextOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if ($context->lane !== TenantInteractionLane::StandardActiveOperating) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::RememberedContextValidity,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::RememberedContextStale,
discoverable: false,
metadata: $this->metadata($context),
);
}
if (! $lifecycle->isSelectableInStandardLane()) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::RememberedContextValidity,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::RememberedContextStale,
discoverable: false,
metadata: $this->metadata($context),
);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::RememberedContextValidity,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
metadata: $this->metadata($context),
);
}
private function tenantBoundViewabilityOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if ($context->lane !== TenantInteractionLane::AdministrativeManagement) {
return $this->wrongLaneOutcome($context, TenantOperabilityQuestion::TenantBoundViewability, $lifecycle);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::TenantBoundViewability,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
metadata: $this->metadata($context),
);
}
private function canonicalViewabilityOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if ($context->lane !== TenantInteractionLane::CanonicalWorkspaceRecord) {
return $this->wrongLaneOutcome($context, TenantOperabilityQuestion::CanonicalLinkedRecordViewability, $lifecycle);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::CanonicalLinkedRecordViewability,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
metadata: $this->metadata($context),
);
}
private function archiveEligibilityOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if ($context->lane !== TenantInteractionLane::AdministrativeManagement) {
return $this->wrongLaneOutcome($context, TenantOperabilityQuestion::ArchiveEligibility, $lifecycle);
}
if (! $lifecycle->canArchive()) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::ArchiveEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: $lifecycle === TenantLifecycle::Archived
? TenantOperabilityReasonCode::TenantAlreadyArchived
: TenantOperabilityReasonCode::SelectorIneligibleLifecycle,
discoverable: true,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::ArchiveEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
private function restoreEligibilityOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if ($context->lane !== TenantInteractionLane::AdministrativeManagement) {
return $this->wrongLaneOutcome($context, TenantOperabilityQuestion::RestoreEligibility, $lifecycle);
}
if (! $lifecycle->canRestore()) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::RestoreEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::TenantNotArchived,
discoverable: true,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::RestoreEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
private function resumeOnboardingOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if (! in_array($context->lane, [TenantInteractionLane::AdministrativeManagement, TenantInteractionLane::OnboardingWorkflow], true)) {
return $this->wrongLaneOutcome($context, TenantOperabilityQuestion::ResumeOnboardingEligibility, $lifecycle);
}
if (! $lifecycle->canResumeOnboarding()) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::ResumeOnboardingEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::SelectorIneligibleLifecycle,
discoverable: $context->lane === TenantInteractionLane::AdministrativeManagement,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
if ($context->onboardingDraft instanceof TenantOnboardingSession && ! $context->onboardingDraft->isWorkflowResumable()) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::ResumeOnboardingEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::OnboardingNotResumable,
discoverable: $context->lane === TenantInteractionLane::AdministrativeManagement,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::ResumeOnboardingEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
private function onboardingCompletionOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if ($context->lane !== TenantInteractionLane::OnboardingWorkflow) {
return $this->wrongLaneOutcome($context, TenantOperabilityQuestion::OnboardingCompletionEligibility, $lifecycle);
}
if (! in_array($lifecycle, [TenantLifecycle::Draft, TenantLifecycle::Onboarding], true)) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::OnboardingCompletionEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::SelectorIneligibleLifecycle,
discoverable: false,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::OnboardingCompletionEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
private function verificationReadinessOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
if (! $lifecycle->supportsQuestion(TenantOperabilityQuestion::VerificationReadinessEligibility, $context->lane)) {
return TenantOperabilityOutcome::deny(
question: TenantOperabilityQuestion::VerificationReadinessEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: $context->lane === TenantInteractionLane::CanonicalWorkspaceRecord
? TenantOperabilityReasonCode::CanonicalViewFollowupOnly
: TenantOperabilityReasonCode::WrongLane,
discoverable: $context->lane === TenantInteractionLane::AdministrativeManagement,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::VerificationReadinessEligibility,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: true,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
private function administrativeDiscoverabilityOutcome(TenantOperabilityContext $context, TenantLifecycle $lifecycle): TenantOperabilityOutcome
{
return TenantOperabilityOutcome::allow(
question: TenantOperabilityQuestion::AdministrativeDiscoverability,
lifecycle: $lifecycle,
lane: $context->lane,
discoverable: $lifecycle->isAdministrativelyDiscoverable(),
metadata: $this->metadata($context),
);
}
private function wrongLaneOutcome(
TenantOperabilityContext $context,
TenantOperabilityQuestion $question,
TenantLifecycle $lifecycle,
): TenantOperabilityOutcome {
return TenantOperabilityOutcome::deny(
question: $question,
lifecycle: $lifecycle,
lane: $context->lane,
reasonCode: TenantOperabilityReasonCode::WrongLane,
discoverable: false,
requiredCapability: $context->requiredCapability,
metadata: $this->metadata($context),
);
}
/**
* @return array<string, mixed>
*/
private function metadata(TenantOperabilityContext $context): array
{
return array_filter([
'selected_tenant_id' => $context->selectedTenant?->getKey(),
'linked_record_type' => $context->linkedRecordType,
'linked_record_id' => $context->linkedRecordId,
'page_category' => $context->pageCategory?->value,
'onboarding_draft_id' => $context->onboardingDraft?->getKey(),
], static fn (mixed $value): bool => $value !== null && $value !== '');
}
}