## Summary - introduce the Provider Connection Filament resource (list/create/edit) with DB-only controls, grouped action dropdowns, and badge-driven status/health rendering - wire up the provider foundation stack (migrations, models, policies, providers, operations, badges, and audits) plus the required spec docs/checklists - standardize Inventory Sync notifications so the job no longer writes its own DB rows; terminal notifications now flow exclusively through OperationRunCompleted while the start surface still shows the queued toast ## Testing - ./vendor/bin/sail php ./vendor/bin/pint --dirty - ./vendor/bin/sail artisan test tests/Unit/Badges/ProviderConnectionBadgesTest.php - ./vendor/bin/sail artisan test tests/Feature/ProviderConnections tests/Feature/Filament/ProviderConnectionsDbOnlyTest.php - ./vendor/bin/sail artisan test tests/Feature/Inventory/RunInventorySyncJobTest.php tests/Feature/Inventory/InventorySyncStartSurfaceTest.php Co-authored-by: Ahmed Darrazi <ahmeddarrazi@MacBookPro.fritz.box> Reviewed-on: #73
68 lines
1.8 KiB
PHP
68 lines
1.8 KiB
PHP
<?php
|
|
|
|
namespace App\Services\Providers;
|
|
|
|
use App\Models\ProviderConnection;
|
|
use App\Services\Graph\GraphResponse;
|
|
use App\Services\Providers\Contracts\ProviderInventoryCollector;
|
|
use RuntimeException;
|
|
|
|
final class MicrosoftProviderInventoryCollector implements ProviderInventoryCollector
|
|
{
|
|
/**
|
|
* @var array<int, string>
|
|
*/
|
|
private array $policyTypes = [
|
|
'deviceConfiguration',
|
|
'settingsCatalogPolicy',
|
|
'groupPolicyConfiguration',
|
|
];
|
|
|
|
public function __construct(private readonly ProviderGateway $gateway) {}
|
|
|
|
public function collect(ProviderConnection $connection): array
|
|
{
|
|
$total = 0;
|
|
|
|
foreach ($this->policyTypes as $policyType) {
|
|
$response = $this->gateway->listPolicies($connection, $policyType);
|
|
|
|
if ($response->failed()) {
|
|
$message = $this->messageForResponse($response);
|
|
$status = (int) ($response->status ?? 0);
|
|
|
|
throw new RuntimeException("Graph request failed for {$policyType} (status {$status}): {$message}");
|
|
}
|
|
|
|
$items = is_array($response->data) ? $response->data : [];
|
|
$total += count($items);
|
|
}
|
|
|
|
return [
|
|
'total' => $total,
|
|
'items' => $total,
|
|
];
|
|
}
|
|
|
|
private function messageForResponse(GraphResponse $response): string
|
|
{
|
|
$error = $response->errors[0] ?? null;
|
|
|
|
if (is_string($error)) {
|
|
return $error;
|
|
}
|
|
|
|
if (is_array($error)) {
|
|
$message = $error['message'] ?? null;
|
|
|
|
if (is_string($message) && $message !== '') {
|
|
return $message;
|
|
}
|
|
|
|
return json_encode($error, JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE) ?: 'Request failed.';
|
|
}
|
|
|
|
return 'Request failed.';
|
|
}
|
|
}
|