Files
vkv/app/Jobs/DuplicateResolutionJob.php
Zdeněk Burda 41e3ce6f25 Initial commit
2026-01-09 21:26:40 +01:00

164 lines
6.1 KiB
PHP

<?php
namespace App\Jobs;
use App\Models\EvaluationRun;
use App\Models\EvaluationRuleSet;
use App\Models\QsoResult;
use App\Models\WorkingQso;
use App\Enums\QsoErrorCode;
use App\Services\Evaluation\EvaluationCoordinator;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Queue\Queueable;
use Throwable;
/**
* Job: DuplicateResolutionJob
*
* WHY:
* - Duplicitní QSO se musí rozhodnout až po matchingu, aby bylo jasné,
* které záznamy jsou spárované a v jakém pořadí.
* ORDER:
* - Spouští se po UnpairedClassificationJob (match je hotový, error_code stabilní).
* - Krok je nevratný: duplicitní QSO jsou označena DUP a další kroky
* už jen počítají body podle policy.
*
* Vstup:
* - WorkingQso (dupe_key per log)
* - QsoResult s error_code/matched_log_qso_id
* - EvaluationRuleSet (dup_resolution_strategy)
*
* Výstup:
* - Nastavení DUP u všech „non-survivor“ QSO
*
* Co job NEDĚLÁ:
* - neprovádí matching protistanic,
* - nepočítá body ani penalizace,
* - neupravuje původní log_qsos.
*/
class DuplicateResolutionJob implements ShouldQueue
{
use Queueable;
public int $tries = 2;
public array $backoff = [60];
public function __construct(
protected int $evaluationRunId
) {
}
public function handle(): void
{
$run = EvaluationRun::find($this->evaluationRunId);
if (! $run || $run->isCanceled()) {
return;
}
$coordinator = new EvaluationCoordinator();
try {
$ruleSet = EvaluationRuleSet::find($run->rule_set_id);
if (! $ruleSet) {
$coordinator->eventError($run, 'Duplicitní QSO nelze vyhodnotit: chybí ruleset.', [
'step' => 'duplicate_resolution',
]);
return;
}
$coordinator->eventInfo($run, 'Duplicate: krok spuštěn.', [
'step' => 'duplicate_resolution',
'round_id' => $run->round_id,
]);
$coordinator->eventInfo($run, 'Detekce duplicitních QSO.', [
'step' => 'match',
'round_id' => $run->round_id,
'step_progress_done' => null,
'step_progress_total' => $run->progress_total,
]);
$strategy = $ruleSet->dupResolutionStrategy();
$working = WorkingQso::where('evaluation_run_id', $run->id)->get();
$byLog = $working->groupBy('log_id');
foreach ($byLog as $logId => $items) {
if (EvaluationRun::isCanceledRun($run->id)) {
return;
}
$byDupeKey = $items->groupBy('dupe_key');
foreach ($byDupeKey as $dupeKey => $dupes) {
if (! $dupeKey || $dupes->count() < 2) {
continue;
}
$sorted = $dupes->sort(function ($a, $b) use ($strategy, $run) {
$resultA = QsoResult::where('evaluation_run_id', $run->id)
->where('log_qso_id', $a->log_qso_id)
->first();
$resultB = QsoResult::where('evaluation_run_id', $run->id)
->where('log_qso_id', $b->log_qso_id)
->first();
foreach ($strategy as $rule) {
if ($rule === 'paired_first') {
$aPaired = $resultA && $resultA->matched_log_qso_id !== null;
$bPaired = $resultB && $resultB->matched_log_qso_id !== null;
if ($aPaired !== $bPaired) {
return $aPaired ? -1 : 1;
}
}
if ($rule === 'ok_first') {
$aOk = $resultA && $resultA->error_code === QsoErrorCode::OK;
$bOk = $resultB && $resultB->error_code === QsoErrorCode::OK;
if ($aOk !== $bOk) {
return $aOk ? -1 : 1;
}
}
if ($rule === 'earlier_time') {
$tsA = $a->ts_utc?->getTimestamp() ?? PHP_INT_MAX;
$tsB = $b->ts_utc?->getTimestamp() ?? PHP_INT_MAX;
if ($tsA !== $tsB) {
return $tsA <=> $tsB;
}
}
if ($rule === 'lower_id') {
if ($a->log_qso_id !== $b->log_qso_id) {
return $a->log_qso_id <=> $b->log_qso_id;
}
}
}
return $a->log_qso_id <=> $b->log_qso_id;
})->values();
$survivor = $sorted->shift();
foreach ($sorted as $dupe) {
QsoResult::where('evaluation_run_id', $run->id)
->where('log_qso_id', $dupe->log_qso_id)
->update([
'is_duplicate' => true,
'is_valid' => false,
'error_code' => QsoErrorCode::DUP,
'error_side' => 'NONE',
]);
}
}
}
EvaluationRun::where('id', $run->id)->increment('progress_done');
$coordinator->eventInfo($run, 'Duplicate: krok dokončen.', [
'step' => 'duplicate_resolution',
'round_id' => $run->round_id,
]);
} catch (Throwable $e) {
$coordinator->eventError($run, 'Duplicate: krok selhal.', [
'step' => 'duplicate_resolution',
'round_id' => $run->round_id,
'error' => $e->getMessage(),
]);
throw $e;
}
}
}