TenantAtlas/apps/platform/tests/Feature/Operations/AssignmentRunSummaryCountsTest.php
ahmido ce0615a9c1 Spec 182: relocate Laravel platform to apps/platform (#213)
## Summary
- move the Laravel application into `apps/platform` and keep the repository root for orchestration, docs, and tooling
- update the local command model, Sail/Docker wiring, runtime paths, and ignore rules around the new platform location
- add relocation quickstart/contracts plus focused smoke coverage for bootstrap, command model, routes, and runtime behavior

## Validation
- `cd apps/platform && ./vendor/bin/sail artisan test --compact tests/Feature/PlatformRelocation`
- integrated browser smoke validated `/up`, `/`, `/admin`, `/admin/choose-workspace`, and tenant route semantics for `200`, `403`, and `404`

## Remaining Rollout Checks
- validate Dokploy build context and working-directory assumptions against the new `apps/platform` layout
- confirm web, queue, and scheduler processes all start from the expected working directory in staging/production
- verify no legacy volume mounts or asset-publish paths still point at the old root-level `public/` or `storage/` locations

Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de>
Reviewed-on: #213
2026-04-08 08:40:47 +00:00

147 lines
4.7 KiB
PHP

<?php
use App\Jobs\FetchAssignmentsJob;
use App\Jobs\RestoreAssignmentsJob;
use App\Models\BackupItem;
use App\Models\BackupSet;
use App\Models\OperationRun;
use App\Models\RestoreRun;
use App\Models\Tenant;
use App\Services\AssignmentBackupService;
use App\Services\AssignmentRestoreService;
use App\Services\OperationRunService;
use Illuminate\Foundation\Testing\RefreshDatabase;
uses(RefreshDatabase::class);
test('fetch assignment job persists terminal summary counts and overwrites previous values', function (): void {
$tenant = Tenant::factory()->create();
$backupSet = BackupSet::factory()->create([
'tenant_id' => (int) $tenant->getKey(),
]);
$backupItem = BackupItem::factory()->create([
'tenant_id' => (int) $tenant->getKey(),
'backup_set_id' => (int) $backupSet->getKey(),
'policy_type' => 'settingsCatalogPolicy',
'policy_identifier' => 'policy-fetch-summary',
'metadata' => [],
]);
$operationRun = OperationRun::factory()->for($tenant)->create([
'workspace_id' => (int) $tenant->workspace_id,
'type' => 'assignments.fetch',
'status' => 'queued',
'outcome' => 'pending',
'summary_counts' => [
'total' => 99,
'processed' => 99,
'failed' => 99,
],
]);
$assignmentBackupService = \Mockery::mock(AssignmentBackupService::class);
$assignmentBackupService->shouldReceive('enrichWithAssignments')
->once()
->andReturnUsing(function (BackupItem $item): BackupItem {
$metadata = is_array($item->metadata) ? $item->metadata : [];
$metadata['assignments_fetch_failed'] = false;
$item->update([
'metadata' => $metadata,
'assignments' => [
['id' => 'assignment-1'],
],
]);
return $item->refresh();
});
$job = new FetchAssignmentsJob(
backupItemId: (int) $backupItem->getKey(),
tenantExternalId: (string) $tenant->external_id,
policyExternalId: (string) $backupItem->policy_identifier,
policyPayload: ['id' => (string) $backupItem->policy_identifier],
operationRun: $operationRun,
);
$job->handle($assignmentBackupService, app(OperationRunService::class));
$operationRun->refresh();
expect($operationRun->status)->toBe('completed')
->and($operationRun->outcome)->toBe('succeeded')
->and($operationRun->summary_counts)->toMatchArray([
'total' => 1,
'processed' => 1,
'failed' => 0,
]);
});
test('restore assignment job persists terminal summary counts and overwrites previous values', function (): void {
$tenant = Tenant::factory()->create();
$backupSet = BackupSet::factory()->create([
'tenant_id' => (int) $tenant->getKey(),
]);
$restoreRun = RestoreRun::factory()->create([
'tenant_id' => (int) $tenant->getKey(),
'backup_set_id' => (int) $backupSet->getKey(),
]);
$operationRun = OperationRun::factory()->for($tenant)->create([
'workspace_id' => (int) $tenant->workspace_id,
'type' => 'assignments.restore',
'status' => 'queued',
'outcome' => 'pending',
'summary_counts' => [
'total' => 88,
'processed' => 88,
'failed' => 88,
],
]);
$assignmentRestoreService = \Mockery::mock(AssignmentRestoreService::class);
$assignmentRestoreService->shouldReceive('restore')
->once()
->andReturn([
'outcomes' => [
['status' => 'success'],
['status' => 'success'],
['status' => 'failed', 'reason' => 'Graph request failed'],
],
'summary' => [
'success' => 2,
'failed' => 1,
'skipped' => 0,
],
]);
$job = new RestoreAssignmentsJob(
restoreRunId: (int) $restoreRun->getKey(),
tenantId: (int) $tenant->getKey(),
policyType: 'settingsCatalogPolicy',
policyId: 'policy-restore-summary',
assignments: [
['id' => 'assignment-1'],
['id' => 'assignment-2'],
['id' => 'assignment-3'],
],
groupMapping: [],
foundationMapping: [],
operationRun: $operationRun,
);
$job->handle($assignmentRestoreService, app(OperationRunService::class));
$operationRun->refresh();
expect($operationRun->status)->toBe('completed')
->and($operationRun->outcome)->toBe('partially_succeeded')
->and($operationRun->summary_counts)->toMatchArray([
'total' => 3,
'processed' => 3,
'failed' => 1,
]);
});