TenantAtlas/apps/platform/tests/Unit/BulkPolicyVersionRestoreJobTest.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

105 lines
3.5 KiB
PHP

<?php
use App\Jobs\BulkPolicyVersionRestoreJob;
use App\Models\OperationRun;
use App\Models\Policy;
use App\Models\PolicyVersion;
use App\Models\Tenant;
use App\Models\User;
use App\Services\OperationRunService;
use Illuminate\Foundation\Testing\RefreshDatabase;
uses(RefreshDatabase::class);
test('bulk policy version restore restores archived versions', function () {
$tenant = Tenant::factory()->create(['is_current' => true]);
$user = User::factory()->create();
$policy = Policy::factory()->create(['tenant_id' => $tenant->id]);
$version = PolicyVersion::factory()->create([
'tenant_id' => $tenant->id,
'policy_id' => $policy->id,
'version_number' => 1,
'captured_at' => now()->subDays(120),
]);
$version->delete();
expect($version->trashed())->toBeTrue();
$run = OperationRun::factory()->create([
'tenant_id' => $tenant->id,
'user_id' => $user->id,
'type' => 'policy_version.restore',
'status' => 'queued',
'outcome' => 'pending',
'summary_counts' => [],
'failure_summary' => [],
'context' => [
'target_scope' => ['directory_context_id' => 1],
],
]);
(new BulkPolicyVersionRestoreJob(
tenantId: $tenant->id,
userId: $user->id,
policyVersionIds: [$version->id],
operationRun: $run,
))->handle(app(OperationRunService::class));
$version->refresh();
expect($version->trashed())->toBeFalse();
$run->refresh();
expect($run->status)->toBe('completed')
->and($run->outcome)->toBe('succeeded')
->and($run->summary_counts['total'] ?? null)->toBe(1)
->and($run->summary_counts['processed'] ?? null)->toBe(1)
->and($run->summary_counts['succeeded'] ?? null)->toBe(1)
->and((int) ($run->summary_counts['skipped'] ?? 0))->toBe(0)
->and((int) ($run->summary_counts['failed'] ?? 0))->toBe(0);
});
test('bulk policy version restore skips active versions', function () {
$tenant = Tenant::factory()->create(['is_current' => true]);
$user = User::factory()->create();
$policy = Policy::factory()->create(['tenant_id' => $tenant->id]);
$version = PolicyVersion::factory()->create([
'tenant_id' => $tenant->id,
'policy_id' => $policy->id,
'version_number' => 1,
'captured_at' => now()->subDays(120),
]);
$run = OperationRun::factory()->create([
'tenant_id' => $tenant->id,
'user_id' => $user->id,
'type' => 'policy_version.restore',
'status' => 'queued',
'outcome' => 'pending',
'summary_counts' => [],
'failure_summary' => [],
'context' => [
'target_scope' => ['directory_context_id' => 1],
],
]);
(new BulkPolicyVersionRestoreJob(
tenantId: $tenant->id,
userId: $user->id,
policyVersionIds: [$version->id],
operationRun: $run,
))->handle(app(OperationRunService::class));
$version->refresh();
expect($version->trashed())->toBeFalse();
$run->refresh();
expect($run->status)->toBe('completed')
->and($run->outcome)->toBe('succeeded')
->and($run->summary_counts['total'] ?? null)->toBe(1)
->and($run->summary_counts['processed'] ?? null)->toBe(1)
->and((int) ($run->summary_counts['succeeded'] ?? 0))->toBe(0)
->and($run->summary_counts['skipped'] ?? null)->toBe(1)
->and((int) ($run->summary_counts['failed'] ?? 0))->toBe(0);
});