TenantAtlas/app/Models/AlertRule.php
Ahmed Darrazi 222a7e0a97 feat(104): implement Provider Permission Posture
- T001-T014: Foundation - StoredReport model/migration, Finding resolved
  lifecycle, badge mappings (resolved status, permission_posture type),
  OperationCatalog + AlertRule constants
- T015-T022: US1 - PermissionPostureFindingGenerator with fingerprint-based
  idempotent upsert, severity from feature-impact count, auto-resolve on
  grant, auto-reopen on revoke, error findings (FR-015), stale finding
  cleanup; GeneratePermissionPostureFindingsJob dispatched from health check;
  PostureResult VO + PostureScoreCalculator
- T023-T026: US2+US4 - Stored report payload validation, temporal ordering,
  polymorphic reusability, score accuracy acceptance tests
- T027-T029: US3 - EvaluateAlertsJob.permissionMissingEvents() wired into
  alert pipeline, AlertRuleResource event type option, cooldown/dedupe tests
- T030-T034: Polish - PruneStoredReportsCommand with config retention,
  scheduled daily, end-to-end integration test, Pint clean

UI bug fixes found during testing:
- FindingResource: hide Diff section for non-drift findings
- TenantRequiredPermissions: fix re-run verification link
- tenant-required-permissions.blade.php: preserve details open state

70 tests (50 PermissionPosture + 20 FindingResolved/Badge/Alert), 216 assertions
2026-02-21 23:31:03 +01:00

68 lines
1.8 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Models;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
use Illuminate\Database\Eloquent\Relations\HasMany;
class AlertRule extends Model
{
use HasFactory;
public const string EVENT_HIGH_DRIFT = 'high_drift';
public const string EVENT_COMPARE_FAILED = 'compare_failed';
public const string EVENT_SLA_DUE = 'sla_due';
public const string EVENT_PERMISSION_MISSING = 'permission_missing';
public const string TENANT_SCOPE_ALL = 'all';
public const string TENANT_SCOPE_ALLOWLIST = 'allowlist';
protected $guarded = [];
protected $casts = [
'is_enabled' => 'boolean',
'tenant_allowlist' => 'array',
'cooldown_seconds' => 'integer',
'quiet_hours_enabled' => 'boolean',
];
public function workspace(): BelongsTo
{
return $this->belongsTo(Workspace::class);
}
public function destinations(): BelongsToMany
{
return $this->belongsToMany(AlertDestination::class, 'alert_rule_destinations')
->using(AlertRuleDestination::class)
->withPivot(['id', 'workspace_id'])
->withTimestamps();
}
public function deliveries(): HasMany
{
return $this->hasMany(AlertDelivery::class);
}
public function appliesToTenant(int $tenantId): bool
{
if ($this->tenant_scope_mode !== self::TENANT_SCOPE_ALLOWLIST) {
return true;
}
$allowlist = is_array($this->tenant_allowlist) ? $this->tenant_allowlist : [];
$allowlist = array_values(array_unique(array_map(static fn (mixed $value): int => (int) $value, $allowlist)));
return in_array($tenantId, $allowlist, true);
}
}