Phase 1: Setup & Database (13 tasks completed) - Add assignments JSONB column to backup_items table - Add group_mapping JSONB column to restore_runs table - Extend BackupItem model with 7 assignment accessor methods - Extend RestoreRun model with 8 group mapping helper methods - Add scopeWithAssignments() query scope to BackupItem - Update graph_contracts.php with assignments endpoints - Create 5 factories: BackupItem, RestoreRun, Tenant, BackupSet, Policy - Add 30 unit tests (15 BackupItem, 15 RestoreRun) - all passing Phase 2: Graph API Integration (16 tasks completed) - Create AssignmentFetcher service with fallback strategy - Create GroupResolver service with orphaned ID handling - Create ScopeTagResolver service with 1-hour caching - Implement fail-soft error handling for all services - Add 17 unit tests (5 AssignmentFetcher, 6 GroupResolver, 6 ScopeTagResolver) - all passing - Total: 71 assertions across all Phase 2 tests Test Results: - Phase 1: 30/30 tests passing (45 assertions) - Phase 2: 17/17 tests passing (71 assertions) - Total: 47/47 tests passing (116 assertions) - Code formatted with Pint (PSR-12 compliant) Next: Phase 3 - US1 Backup with Assignments (12 tasks)
158 lines
4.8 KiB
PHP
158 lines
4.8 KiB
PHP
<?php
|
|
|
|
use App\Models\BackupItem;
|
|
use Illuminate\Foundation\Testing\RefreshDatabase;
|
|
use Tests\TestCase;
|
|
|
|
uses(TestCase::class, RefreshDatabase::class);
|
|
|
|
test('assignments cast works', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'assignments' => [
|
|
['id' => 'abc-123', 'target' => ['groupId' => 'group-1']],
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->assignments)->toBeArray()
|
|
->and($backupItem->assignments)->toHaveCount(1);
|
|
});
|
|
|
|
test('getAssignmentCountAttribute returns correct count', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'assignments' => [
|
|
['id' => 'abc-123', 'target' => ['groupId' => 'group-1']],
|
|
['id' => 'def-456', 'target' => ['groupId' => 'group-2']],
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->assignment_count)->toBe(2);
|
|
});
|
|
|
|
test('getAssignmentCountAttribute returns zero for null assignments', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'assignments' => null,
|
|
]);
|
|
|
|
expect($backupItem->assignment_count)->toBe(0);
|
|
});
|
|
|
|
test('hasAssignments returns true when assignments exist', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'assignments' => [
|
|
['id' => 'abc-123', 'target' => ['groupId' => 'group-1']],
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->hasAssignments())->toBeTrue();
|
|
});
|
|
|
|
test('hasAssignments returns false when assignments are null', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'assignments' => null,
|
|
]);
|
|
|
|
expect($backupItem->hasAssignments())->toBeFalse();
|
|
});
|
|
|
|
test('getGroupIdsAttribute extracts unique group IDs', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'assignments' => [
|
|
['id' => 'abc-123', 'target' => ['groupId' => 'group-1']],
|
|
['id' => 'def-456', 'target' => ['groupId' => 'group-2']],
|
|
['id' => 'ghi-789', 'target' => ['groupId' => 'group-1']], // duplicate
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->group_ids)->toHaveCount(2)
|
|
->and($backupItem->group_ids)->toContain('group-1', 'group-2');
|
|
});
|
|
|
|
test('getScopeTagIdsAttribute returns scope tag IDs from metadata', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [
|
|
'scope_tag_ids' => ['0', 'abc-123', 'def-456'],
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->scope_tag_ids)->toHaveCount(3)
|
|
->and($backupItem->scope_tag_ids)->toContain('0', 'abc-123', 'def-456');
|
|
});
|
|
|
|
test('getScopeTagIdsAttribute returns default when not in metadata', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [],
|
|
]);
|
|
|
|
expect($backupItem->scope_tag_ids)->toBe(['0']);
|
|
});
|
|
|
|
test('getScopeTagNamesAttribute returns scope tag names from metadata', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [
|
|
'scope_tag_names' => ['Default', 'HR-Admins', 'Finance'],
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->scope_tag_names)->toHaveCount(3)
|
|
->and($backupItem->scope_tag_names)->toContain('Default', 'HR-Admins', 'Finance');
|
|
});
|
|
|
|
test('getScopeTagNamesAttribute returns default when not in metadata', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [],
|
|
]);
|
|
|
|
expect($backupItem->scope_tag_names)->toBe(['Default']);
|
|
});
|
|
|
|
test('hasOrphanedAssignments returns true when flag is set', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [
|
|
'has_orphaned_assignments' => true,
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->hasOrphanedAssignments())->toBeTrue();
|
|
});
|
|
|
|
test('hasOrphanedAssignments returns false when flag is not set', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [],
|
|
]);
|
|
|
|
expect($backupItem->hasOrphanedAssignments())->toBeFalse();
|
|
});
|
|
|
|
test('assignmentsFetchFailed returns true when flag is set', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [
|
|
'assignments_fetch_failed' => true,
|
|
],
|
|
]);
|
|
|
|
expect($backupItem->assignmentsFetchFailed())->toBeTrue();
|
|
});
|
|
|
|
test('assignmentsFetchFailed returns false when flag is not set', function () {
|
|
$backupItem = BackupItem::factory()->create([
|
|
'metadata' => [],
|
|
]);
|
|
|
|
expect($backupItem->assignmentsFetchFailed())->toBeFalse();
|
|
});
|
|
|
|
test('scopeWithAssignments filters items with assignments', function () {
|
|
BackupItem::factory()->create(['assignments' => null]);
|
|
BackupItem::factory()->create(['assignments' => []]);
|
|
$withAssignments = BackupItem::factory()->create([
|
|
'assignments' => [
|
|
['id' => 'abc-123', 'target' => ['groupId' => 'group-1']],
|
|
],
|
|
]);
|
|
|
|
$result = BackupItem::withAssignments()->get();
|
|
|
|
expect($result)->toHaveCount(1)
|
|
->and($result->first()->id)->toBe($withAssignments->id);
|
|
});
|