Implements the Settings foundation workspace controls. Includes: - Settings foundation UI/controls scoped to workspace context - Related onboarding/consent flow adjustments as included in branch history Testing: - `vendor/bin/sail artisan test --compact --no-ansi --filter=SettingsFoundation` Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #119
92 lines
2.9 KiB
PHP
92 lines
2.9 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
use App\Jobs\ApplyBackupScheduleRetentionJob;
|
|
use App\Models\BackupSchedule;
|
|
use App\Models\BackupSet;
|
|
use App\Models\OperationRun;
|
|
use App\Models\WorkspaceSetting;
|
|
|
|
it('uses workspace retention default when schedule retention is null', function (): void {
|
|
[$user, $tenant] = createUserWithTenant(role: 'manager');
|
|
|
|
WorkspaceSetting::factory()->create([
|
|
'workspace_id' => (int) $tenant->workspace_id,
|
|
'domain' => 'backup',
|
|
'key' => 'retention_keep_last_default',
|
|
'value' => 2,
|
|
'updated_by_user_id' => (int) $user->getKey(),
|
|
]);
|
|
|
|
$schedule = BackupSchedule::query()->create([
|
|
'tenant_id' => (int) $tenant->getKey(),
|
|
'name' => 'Nightly fallback',
|
|
'is_enabled' => true,
|
|
'timezone' => 'UTC',
|
|
'frequency' => 'daily',
|
|
'time_of_day' => '01:00:00',
|
|
'days_of_week' => null,
|
|
'policy_types' => ['deviceConfiguration'],
|
|
'include_foundations' => true,
|
|
'retention_keep_last' => null,
|
|
]);
|
|
|
|
$sets = collect(range(1, 5))->map(function (int $index) use ($tenant): BackupSet {
|
|
return BackupSet::query()->create([
|
|
'tenant_id' => (int) $tenant->getKey(),
|
|
'name' => 'Set '.$index,
|
|
'status' => 'completed',
|
|
'item_count' => 0,
|
|
'completed_at' => now()->subMinutes(20 - $index),
|
|
]);
|
|
});
|
|
|
|
$completedAt = now('UTC')->startOfMinute()->subMinutes(10);
|
|
|
|
foreach ($sets as $set) {
|
|
OperationRun::query()->create([
|
|
'workspace_id' => (int) $tenant->workspace_id,
|
|
'tenant_id' => (int) $tenant->getKey(),
|
|
'user_id' => null,
|
|
'initiator_name' => 'System',
|
|
'type' => 'backup_schedule_run',
|
|
'status' => 'completed',
|
|
'outcome' => 'succeeded',
|
|
'run_identity_hash' => hash('sha256', 'retention-fallback:'.$schedule->id.':'.$set->id),
|
|
'summary_counts' => [
|
|
'total' => 0,
|
|
'processed' => 0,
|
|
'succeeded' => 0,
|
|
],
|
|
'failure_summary' => [],
|
|
'context' => [
|
|
'backup_schedule_id' => (int) $schedule->getKey(),
|
|
'backup_set_id' => (int) $set->getKey(),
|
|
],
|
|
'started_at' => $completedAt,
|
|
'completed_at' => $completedAt,
|
|
]);
|
|
|
|
$completedAt = $completedAt->addMinute();
|
|
}
|
|
|
|
ApplyBackupScheduleRetentionJob::dispatchSync((int) $schedule->getKey());
|
|
|
|
$kept = $sets->take(-2);
|
|
$deleted = $sets->take(3);
|
|
|
|
foreach ($kept as $set) {
|
|
$this->assertDatabaseHas('backup_sets', [
|
|
'id' => (int) $set->getKey(),
|
|
'deleted_at' => null,
|
|
]);
|
|
}
|
|
|
|
foreach ($deleted as $set) {
|
|
$this->assertSoftDeleted('backup_sets', [
|
|
'id' => (int) $set->getKey(),
|
|
]);
|
|
}
|
|
});
|