## Summary - turn the Monitoring audit log placeholder into a real workspace-scoped audit review surface - introduce a shared audit recorder, richer audit value objects, and additive audit log schema evolution - add audit outcome and actor badges, permission-aware related navigation, and durable audit retention coverage ## Included - canonical `/admin/audit-log` list and detail inspection UI - audit model helpers, taxonomy expansion, actor/target snapshots, and recorder/builder services - operation terminal audit writes and purge command retention changes - spec 134 design artifacts and focused Pest coverage for audit foundation behavior ## Validation - `vendor/bin/sail bin pint --dirty --format agent` - `vendor/bin/sail artisan test --compact tests/Unit/Audit tests/Unit/Badges/AuditBadgesTest.php tests/Feature/Filament/AuditLogPageTest.php tests/Feature/Filament/AuditLogDetailInspectionTest.php tests/Feature/Filament/AuditLogAuthorizationTest.php tests/Feature/Monitoring/AuditCoverageGovernanceTest.php tests/Feature/Monitoring/AuditCoverageOperationsTest.php tests/Feature/Console/TenantpilotPurgeNonPersistentDataTest.php` ## Notes - Livewire v4.0+ compliance is preserved within the existing Filament v5 application. - No provider registration changes were needed; panel provider registration remains in `bootstrap/providers.php`. - No new globally searchable resource was introduced. - The audit page remains read-only; no destructive actions were added. - No new asset pipeline changes were introduced; existing deploy-time `php artisan filament:assets` behavior remains unchanged. Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #163
45 lines
1.3 KiB
PHP
45 lines
1.3 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Support\Audit;
|
|
|
|
enum AuditOutcome: string
|
|
{
|
|
case Success = 'success';
|
|
case Failed = 'failed';
|
|
case Partial = 'partial';
|
|
case Info = 'info';
|
|
case Blocked = 'blocked';
|
|
|
|
public static function normalize(mixed $value): self
|
|
{
|
|
$normalized = is_string($value) ? strtolower(trim($value)) : null;
|
|
|
|
return match ($normalized) {
|
|
self::Success->value, 'succeeded', 'completed', 'complete', 'ok' => self::Success,
|
|
self::Failed->value, 'failure', 'error', 'errored' => self::Failed,
|
|
self::Partial->value, 'partially_succeeded', 'partial_success', 'partial_failure' => self::Partial,
|
|
self::Blocked->value, 'skipped', 'deferred', 'cancelled', 'canceled' => self::Blocked,
|
|
self::Info->value, 'pending', 'queued', 'running' => self::Info,
|
|
default => self::Info,
|
|
};
|
|
}
|
|
|
|
public static function normalizeValue(mixed $value): string
|
|
{
|
|
return self::normalize($value)->value;
|
|
}
|
|
|
|
public function label(): string
|
|
{
|
|
return match ($this) {
|
|
self::Success => 'Success',
|
|
self::Failed => 'Failed',
|
|
self::Partial => 'Partial',
|
|
self::Info => 'Info',
|
|
self::Blocked => 'Blocked',
|
|
};
|
|
}
|
|
}
|