From ff671d8d4a4cedd5a5750d895983f70bd4f3748d Mon Sep 17 00:00:00 2001 From: ahmido Date: Sat, 7 Feb 2026 09:18:00 +0000 Subject: [PATCH] fix(spec-079): allow non-UUID ids in inventory_links (#96) ## Why Some Microsoft Graph / Intune identifiers are not UUIDs (e.g. scope tag id "0"). With `inventory_links.source_id` / `target_id` typed as `uuid`, PostgreSQL fails when inventory dependency extraction tries to persist those edges. ## What - PostgreSQL migration changes `inventory_links.source_id` and `inventory_links.target_id` to `text`. - Regression test ensures a non-UUID id ("0") can be persisted; on pgsql it also asserts the columns are `text`. ## Notes - UUID identifiers continue to work (stored as strings). - No UI/Filament changes. ## Testing - `vendor/bin/sail artisan test --compact tests/Feature/Inventory/InventoryLinksNonUuidIdsTest.php` Co-authored-by: Ahmed Darrazi Reviewed-on: https://git.cloudarix.de/ahmido/TenantAtlas/pulls/96 --- ...2108_alter_inventory_links_ids_to_text.php | 47 +++++++++++++++++++ .../079-inventory-links-non-uuid-ids/plan.md | 19 ++++++++ .../079-inventory-links-non-uuid-ids/spec.md | 22 +++++++++ .../079-inventory-links-non-uuid-ids/tasks.md | 18 +++++++ .../InventoryLinksNonUuidIdsTest.php | 46 ++++++++++++++++++ 5 files changed, 152 insertions(+) create mode 100644 database/migrations/2026_02_07_002108_alter_inventory_links_ids_to_text.php create mode 100644 specs/079-inventory-links-non-uuid-ids/plan.md create mode 100644 specs/079-inventory-links-non-uuid-ids/spec.md create mode 100644 specs/079-inventory-links-non-uuid-ids/tasks.md create mode 100644 tests/Feature/Inventory/InventoryLinksNonUuidIdsTest.php diff --git a/database/migrations/2026_02_07_002108_alter_inventory_links_ids_to_text.php b/database/migrations/2026_02_07_002108_alter_inventory_links_ids_to_text.php new file mode 100644 index 0000000..78aba0d --- /dev/null +++ b/database/migrations/2026_02_07_002108_alter_inventory_links_ids_to_text.php @@ -0,0 +1,47 @@ +create(); + $item = InventoryItem::factory()->for($tenant)->create([ + 'external_id' => '11111111-1111-1111-1111-111111111111', + ]); + + /** @var DependencyExtractionService $service */ + $service = app(DependencyExtractionService::class); + + $service->extractForPolicyData($item, [ + 'id' => $item->external_id, + 'roleScopeTagIds' => ['0'], + 'assignments' => [], + ]); + + if ($driver === 'pgsql') { + $columnTypes = collect(DB::select( + "select column_name, data_type from information_schema.columns where table_name = 'inventory_links' and column_name in ('source_id', 'target_id')" + )) + ->mapWithKeys(fn (object $row) => [(string) $row->column_name => (string) $row->data_type]); + + expect($columnTypes->get('source_id'))->toBe('text') + ->and($columnTypes->get('target_id'))->toBe('text'); + } + + expect( + InventoryLink::query() + ->where('tenant_id', $tenant->getKey()) + ->where('source_type', 'inventory_item') + ->where('source_id', $item->external_id) + ->where('relationship_type', RelationshipType::ScopedBy->value) + ->where('target_id', '0') + ->exists() + )->toBeTrue(); +});