$context */ public function __construct( public int $tenantId, public int $userId, public int $baselineRunId, public int $currentRunId, public string $scopeKey, ?OperationRun $operationRun = null, public array $context = [], ) { $this->operationRun = $operationRun; } public function handle( DriftFindingGenerator $generator, OperationRunService $runs, TargetScopeConcurrencyLimiter $limiter, ): void { Log::info('GenerateDriftFindingsJob: started', [ 'tenant_id' => $this->tenantId, 'baseline_run_id' => $this->baselineRunId, 'current_run_id' => $this->currentRunId, 'scope_key' => $this->scopeKey, ]); if (! $this->operationRun instanceof OperationRun) { throw new RuntimeException('OperationRun is required for drift generation.'); } $this->operationRun->refresh(); if ($this->operationRun->status === 'completed') { return; } $opContext = is_array($this->operationRun->context) ? $this->operationRun->context : []; $targetScope = is_array($opContext['target_scope'] ?? null) ? $opContext['target_scope'] : []; $lock = $limiter->acquireSlot($this->tenantId, $targetScope); if (! $lock) { $delay = (int) config('tenantpilot.bulk_operations.poll_interval_seconds', 3); $this->release(max(1, $delay)); return; } try { $tenant = Tenant::query()->find($this->tenantId); if (! $tenant instanceof Tenant) { throw new RuntimeException('Tenant not found.'); } $baseline = InventorySyncRun::query()->find($this->baselineRunId); if (! $baseline instanceof InventorySyncRun) { throw new RuntimeException('Baseline run not found.'); } $current = InventorySyncRun::query()->find($this->currentRunId); if (! $current instanceof InventorySyncRun) { throw new RuntimeException('Current run not found.'); } $runs->updateRun($this->operationRun, 'running'); $counts = is_array($this->operationRun->summary_counts ?? null) ? $this->operationRun->summary_counts : []; if ((int) ($counts['total'] ?? 0) === 0) { $runs->incrementSummaryCounts($this->operationRun, ['total' => 1]); } $created = $generator->generate( tenant: $tenant, baseline: $baseline, current: $current, scopeKey: $this->scopeKey, ); Log::info('GenerateDriftFindingsJob: completed', [ 'tenant_id' => $this->tenantId, 'baseline_run_id' => $this->baselineRunId, 'current_run_id' => $this->currentRunId, 'scope_key' => $this->scopeKey, 'created_findings_count' => $created, ]); $runs->incrementSummaryCounts($this->operationRun, [ 'processed' => 1, 'succeeded' => 1, 'created' => $created, ]); $runs->maybeCompleteBulkRun($this->operationRun); } catch (Throwable $e) { Log::error('GenerateDriftFindingsJob: failed', [ 'tenant_id' => $this->tenantId, 'baseline_run_id' => $this->baselineRunId, 'current_run_id' => $this->currentRunId, 'scope_key' => $this->scopeKey, 'error' => $e->getMessage(), ]); $runs->incrementSummaryCounts($this->operationRun, [ 'processed' => 1, 'failed' => 1, ]); $runs->appendFailures($this->operationRun, [[ 'code' => 'drift.generate.failed', 'message' => $e->getMessage(), ]]); $runs->maybeCompleteBulkRun($this->operationRun); throw $e; } finally { $lock->release(); } } }