## Summary - add canonical managed-tenant onboarding draft routing with explicit draft identity and landing vs concrete draft behavior - implement draft lifecycle, authorization, attribution, picker UX, resume-stage resolution, and auditable cancel or completion semantics - add focused feature, unit, and browser coverage plus Spec 138 artifacts for the onboarding draft resume flow ## Validation - `vendor/bin/sail artisan test --compact tests/Feature/ManagedTenantOnboardingWizardTest.php tests/Feature/Audit/OnboardingDraftAuditTest.php tests/Feature/Onboarding/OnboardingDraftAccessTest.php tests/Feature/Onboarding/OnboardingDraftAuthorizationTest.php tests/Feature/Onboarding/OnboardingDraftLifecycleTest.php tests/Feature/Onboarding/OnboardingDraftMultiTabTest.php tests/Feature/Onboarding/OnboardingDraftPickerTest.php tests/Feature/Onboarding/OnboardingDraftRoutingTest.php tests/Feature/Onboarding/OnboardingRbacSemanticsTest.php tests/Feature/Onboarding/OnboardingVerificationClustersTest.php tests/Feature/Onboarding/OnboardingVerificationTest.php tests/Feature/Onboarding/OnboardingVerificationV1_5UxTest.php tests/Feature/Verification/VerificationReportViewerDbOnlyTest.php tests/Unit/Onboarding tests/Unit/VerificationReportSanitizerEvidenceKindsTest.php tests/Browser/OnboardingDraftRefreshTest.php tests/Browser/OnboardingDraftVerificationResumeTest.php` - passed: 69 tests, 251 assertions Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #167
68 lines
2.0 KiB
PHP
68 lines
2.0 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Services\Onboarding;
|
|
|
|
use App\Models\TenantOnboardingSession;
|
|
use App\Models\User;
|
|
use App\Models\Workspace;
|
|
use Illuminate\Auth\Access\AuthorizationException;
|
|
use Illuminate\Database\Eloquent\Collection;
|
|
use Illuminate\Support\Facades\Gate;
|
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
|
|
|
class OnboardingDraftResolver
|
|
{
|
|
/**
|
|
* @throws AuthorizationException
|
|
* @throws NotFoundHttpException
|
|
*/
|
|
public function resolve(TenantOnboardingSession|int|string $draft, User $user, Workspace $workspace): TenantOnboardingSession
|
|
{
|
|
$draftId = $draft instanceof TenantOnboardingSession
|
|
? (int) $draft->getKey()
|
|
: (int) $draft;
|
|
|
|
$resolvedDraft = TenantOnboardingSession::query()
|
|
->with(['tenant', 'startedByUser', 'updatedByUser'])
|
|
->whereKey($draftId)
|
|
->first();
|
|
|
|
if (! $resolvedDraft instanceof TenantOnboardingSession) {
|
|
throw new NotFoundHttpException;
|
|
}
|
|
|
|
if ((int) $resolvedDraft->workspace_id !== (int) $workspace->getKey()) {
|
|
throw new NotFoundHttpException;
|
|
}
|
|
|
|
Gate::forUser($user)->authorize('view', $resolvedDraft);
|
|
|
|
return $resolvedDraft;
|
|
}
|
|
|
|
/**
|
|
* @return Collection<int, TenantOnboardingSession>
|
|
*/
|
|
public function resumableDraftsFor(User $user, Workspace $workspace): Collection
|
|
{
|
|
$drafts = TenantOnboardingSession::query()
|
|
->with(['tenant', 'startedByUser', 'updatedByUser'])
|
|
->where('workspace_id', (int) $workspace->getKey())
|
|
->resumable()
|
|
->orderByDesc('updated_at')
|
|
->get();
|
|
|
|
return $drafts->filter(function (TenantOnboardingSession $draft) use ($user): bool {
|
|
try {
|
|
Gate::forUser($user)->authorize('view', $draft);
|
|
} catch (AuthorizationException) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
})->values();
|
|
}
|
|
}
|