Re-import zamowienia wykrywa tranzycje payment_status 0/1->2 i emituje payment.status_changed, dzieki czemu chain reguly automatyzacji #7 zmienia status na w_realizacji. - OrderImportRepository: rozdzielenie paymentTransition (event 0/1->2) i statusOverwriteAllowed (preservacja status_code z Phase 62) - AllegroOrderImportService + ShopproOrdersSyncService: emit payment.status_changed na re-imporcie (gate !wasCreated && wasPaymentTransition) - bin/backfill_payment_transition_111.php: jednorazowy CLI dla zamowien payment_status=2 && status_code='nieoplacone' (Allegro + shopPRO) - Naprawa luki znalezionej w analizie zamowienia #864 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
19 KiB
phase, plan, type, wave, depends_on, files_modified, autonomous, delegation
| phase | plan | type | wave | depends_on | files_modified | autonomous | delegation | ||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| 111-payment-transition-event | 01 | execute | 1 |
|
true | off |
Purpose
Naprawa luki obserwowanej dla zamowienia #864 (Allegro): zamowienie zaimportowane 10s po zlozeniu, gdy Allegro jeszcze nie potwierdzilo platnosci (payment_status=0). Re-import 2 minuty pozniej zaktualizowal payment_status do 2, ale order.imported jest gated przez $wasCreated (commit 7eefd1a, Phase 98), wiec automatyzacja "Zmien status na w realizacji (allegro)" nigdy nie odpalila. Allegro nie ma odpowiednika ShopproPaymentStatusSyncService, wiec tranzycja platnosci znika cicho. ShopPRO ma identyczna luke w ShopproOrdersSyncService (flaga payment_transition jest wykrywana w repo, ale nie emituje eventu).
Output
OrderImportRepository::upsertOrderAggregatezwracapayment_transition=truegdy poprzednipayment_statusbyl 0 lub 1 i nowy to 2 (rozszerzenie istniejacej logiki opartej tylko ostatus_code='nieoplacone')- AllegroOrderImportService i ShopproOrdersSyncService emituja
automationService->trigger('payment.status_changed', ...)napayment_transition - CLI
bin/backfill_payment_transition_111.phpnaprawia istniejace zamowienia zpayment_status=2 && status_code='nieoplacone' - Aktualizacja
.paul/codebase/architecture.md(sekcja Order Lifecycle) i.paul/codebase/tech_changelog.md
Project Context
@.paul/PROJECT.md @.paul/STATE.md @.paul/ROADMAP.md
Codebase Refs
@.paul/codebase/architecture.md @.paul/codebase/db_schema.md
Source Files
@src/Modules/Orders/OrderImportRepository.php @src/Modules/Settings/AllegroOrderImportService.php @src/Modules/Settings/ShopproOrdersSyncService.php @src/Modules/Automation/AutomationService.php @bin/backfill_shipped_status_98.php
Reference: Existing Pattern
- ShopproPaymentStatusSyncService.php:256 —
$this->automation?->trigger('payment.status_changed', $orderId, [...])— wzor invocation - OrderImportRepository.php:41-50 — istniejaca logika
paymentTransition(status-based, do rozszerzenia) - bin/backfill_shipped_status_98.php — wzor jednorazowego CLI backfillu
- Regula #7 w DB (sprawdzona, payment_status=2, akcja update_order_status → w_realizacji, BEZ warunku integration_id) — odpali sie dla Allegro automatycznie po wyemitowaniu eventu
<acceptance_criteria>
AC-1: Repo wykrywa tranzycje 0/1 → 2
Given zamowienie istnieje w DB z payment_status=0 (lub 1) i status_code='nieoplacone'
When AllegroOrderImportService::importSingleOrder lub ShopproOrdersSyncService re-importuje to zamowienie z payment_status=2 w aggregate
Then OrderImportRepository::upsertOrderAggregate zwraca `payment_transition=true`
And status_code w bazie zostaje zaktualizowany z payloadu (nie zachowany jako nieoplacone)
Given zamowienie istnieje w DB z payment_status=2
When re-import wykona aggregate z payment_status=2
Then `payment_transition=false` (brak duplikatu eventu przy kolejnych re-importach)
Given zamowienie istnieje w DB z payment_status=0 i status_code='anulowane' (lub inny niz nieoplacone)
When re-import wykona aggregate z payment_status=2
Then `payment_transition=true` (wykrywanie wg payment_status, nie status_code)
And status_code zostaje chroniony zgodnie z istniejaca logika preservacji (Phase 62) — pole status_code nie jest nadpisywane jesli currentStatus != 'nieoplacone'
AC-2: Allegro i shopPRO re-import emituje payment.status_changed
Given OrderImportRepository zwrocilo `payment_transition=true` z `created=false`
When AllegroOrderImportService::importSingleOrder lub ShopproOrdersSyncService::importOneOrder kontynuuje
Then automationService->trigger('payment.status_changed', $orderId, $context) zostaje wywolane
And $context zawiera: integration_id, source ('allegro' lub 'shoppro'), new_payment_status='2', old_payment_status (numeric string z DB)
And automation_execution_logs ma wpis dla rule_id=7 (Zmien status na oplacone) ze status='success'
And status_code zamowienia w DB zmienia sie z 'nieoplacone' na 'w_realizacji' (przez chain reguly 7)
AC-3: Backfill naprawia historyczne dane
Given w DB istnieja zamowienia z payment_status=2 AND LOWER(status_code)='nieoplacone' AND source IN ('allegro', 'shoppro')
When operator uruchamia `php bin/backfill_payment_transition_111.php`
Then kazde takie zamowienie dostaje update status_code = 'w_realizacji' przez OrdersRepository::updateOrderStatus
And order_status_history dostaje wpis nieoplacone -> w_realizacji z change_source='import' i actor='Backfill 111'
And skrypt loguje na stdout liste zamowien (id, source, integration_id, internal_order_number)
And ponowne uruchomienie skryptu jest no-opem (idempotencja: po pierwszym przebiegu zadne zamowienie nie spelnia warunku)
AC-4: Brak regresji w istniejacym flow
Given pierwszy import zamowienia (created=true) z payment_status=2
When AllegroOrderImportService::importSingleOrder konczy
Then `order.imported` jest emitowane jak dotychczas (gate $wasCreated)
And `payment.status_changed` NIE jest emitowane przy created=true (transition liczy sie tylko przy update)
Given zamowienie shopPRO juz oplacone w DB (payment_status=2)
When osobny ShopproPaymentStatusSyncService cron wykryje brak zmian
Then nowy emit z importu nie powiela starego — istniejaca regula 7 idempotentnie zostawia status w_realizacji bez zmian (status_code juz jest w_realizacji, brak nowego status_history wpisu)
</acceptance_criteria>
Task 1: Rozszerz OrderImportRepository o detekcje tranzycji 0/1 → 2 src/Modules/Orders/OrderImportRepository.php 1. W `upsertOrderAggregate` (linie 33-81), w bloku `if (!$created)`: - Zamiast `getCurrentStatus($existingOrderId)`, wprowadz `getCurrentStatusAndPaymentStatus(int $orderId): array{status_code:string, payment_status:int}` ktora jednym zapytaniem czyta oba pola. - Wylicz `$paymentTransition = in_array($oldPaymentStatus, [0, 1], true) && $newPaymentStatus === 2;` - Zachowaj istniejaca logike preservacji status_code: `$statusOverwriteAllowed = ($currentStatus === 'nieoplacone' && $newPaymentStatus === 2);` — to (a nie paymentTransition) decyduje czy `$orderData['status_code']` zostaje nadpisany (Phase 62). - W `replacePayments`/`replaceShipments`/`replaceStatusHistory` warunek bedzie teraz `$paymentTransition || $statusOverwriteAllowed` — utrzymujemy obecne zachowanie shopPRO (replacePayments przy paymentTransition). 2. Wartosc zwracana: zachowaj `[order_id, created, payment_transition]` ale teraz `payment_transition` ma szersza semantyke (0/1→2 niezalznie od status_code). 3. Dodaj `private function getCurrentStatusAndPaymentStatus(int $orderId): array`. Stary `getCurrentStatus()` mozna usunac jesli nie uzywany w innym miejscu (sprawdzic grep).Uwaga:
- NIE zmieniaj logiki `replaceStatusHistory($orderId, ...)` — historia tworzona tylko przy `$created` (zachowane w nowym warunku przy `$statusOverwriteAllowed`). Jesli payment_transition NIE rowna sie statusOverwriteAllowed, paymetTransition NIE wymusza nowego wpisu historii (chain reguly 7 wpisze go przez OrdersRepository::updateOrderStatus).
- `php -l src/Modules/Orders/OrderImportRepository.php` (lint OK)
- Manualny test SQL: `SELECT payment_status, status_code FROM orders WHERE id = 864;` przed re-importem testowego (lokalna kopia) → 0/nieoplacone, po wymuszonym re-imporcie powinno byc 2/(zachowany lub w_realizacji).
- Brak zmian w testach jednostkowych jesli nie istnieja dla tego repo (sprawdzic `tests/Unit/`).
AC-1 spelnione: payment_transition oparte o porownanie payment_status, status_code chroniony zgodnie z dotychczasowa logika preservacji.
Task 2: Emit payment.status_changed w Allegro + shopPRO import services
src/Modules/Settings/AllegroOrderImportService.php, src/Modules/Settings/ShopproOrdersSyncService.php
AllegroOrderImportService.php (po linii 99-106, po istniejacym blocku `order.imported`):
```php
$wasPaymentTransition = !empty($saveResult['payment_transition']);
if (!$wasCreated && $wasPaymentTransition && $this->automationService !== null) {
$this->automationService->trigger('payment.status_changed', $savedOrderId, [
'source' => IntegrationSources::ALLEGRO,
'integration_id' => (int) ($mapped['order']['integration_id'] ?? 0),
'old_payment_status' => '', // nieznane na poziomie service; repo nie eksponuje, nie kluczowe dla rule 7 (sprawdza tylko new_payment_status)
'new_payment_status' => (string) ($mapped['order']['payment_status'] ?? ''),
]);
}
```
ShopproOrdersSyncService.php (po linii 273-280, po istniejacym blocku `order.imported`):
```php
$wasPaymentTransition = !empty($save['payment_transition']);
if (!$wasCreated && $wasPaymentTransition && $this->automationService !== null) {
$this->automationService->trigger('payment.status_changed', $savedOrderId, [
'source' => 'shoppro',
'integration_id' => $integrationId,
'old_payment_status' => '',
'new_payment_status' => (string) ($aggregate['order']['payment_status'] ?? ''),
]);
}
```
Uwagi:
- `order.imported` musi pozostac gated przez `$wasCreated` — NIE zmieniaj tego (commit 7eefd1a).
- Emit `payment.status_changed` jest wykonywany TYLKO gdy `!$wasCreated` (re-import) — przy pierwszym imporcie payment_transition jest false (created=true → nie wchodzi do bloku detekcji).
- Sciezka `status_sync` (AllegroStatusSyncService → importSingleOrder('status_sync')) korzysta z tej samej metody, wiec automatycznie skorzysta z nowego emit.
- Reuse: dolacz brakujacy use `App\Core\Constants\IntegrationSources` jesli jeszcze nie ma (sprawdzic).
- `php -l src/Modules/Settings/AllegroOrderImportService.php`
- `php -l src/Modules/Settings/ShopproOrdersSyncService.php`
- Lokalny test: rezstart importu zamowienia testowego z payment_status=0 → ustaw payment.status w Allegro → kolejny status_sync → sprawdz `automation_execution_logs WHERE event_type='payment.status_changed' AND order_id=...` (powinien byc wpis dla rule_id=7) i `orders.status_code` powinno byc 'w_realizacji'.
AC-2 spelnione: oba serwisy importu emituja `payment.status_changed` na payment_transition. AC-4 zachowane: order.imported nadal gated.
Task 3: Backfill CLI dla istniejacych zamowien z luka platnosci
bin/backfill_payment_transition_111.php
Stworz nowy plik wzorem `bin/backfill_shipped_status_98.php`:
```php
<?php
declare(strict_types=1);
require __DIR__ . '/../bootstrap/app.php';
/** @var App\Core\Application $app */
$app = require __DIR__ . '/../bootstrap/app.php';
$pdo = $app->pdo();
$orders = $app->orders();
$statement = $pdo->prepare(
"SELECT id, internal_order_number, source, integration_id
FROM orders
WHERE source IN ('allegro', 'shoppro')
AND payment_status = 2
AND LOWER(COALESCE(status_code, '')) = 'nieoplacone'"
);
$statement->execute();
$rows = $statement->fetchAll(PDO::FETCH_ASSOC) ?: [];
if ($rows === []) {
echo "[backfill-111] Brak zamowien do naprawy.\n";
exit(0);
}
echo "[backfill-111] Znaleziono " . count($rows) . " zamowien:\n";
foreach ($rows as $row) {
$orderId = (int) $row['id'];
$internalNumber = (string) $row['internal_order_number'];
$source = (string) $row['source'];
$integrationId = (int) $row['integration_id'];
$updated = $orders->updateOrderStatus(
$orderId,
'w_realizacji',
'import',
'Backfill 111'
);
$marker = $updated ? 'OK' : 'SKIP';
echo sprintf(" [%s] #%d %s (source=%s, integration=%d)\n", $marker, $orderId, $internalNumber, $source, $integrationId);
}
echo "[backfill-111] Zakonczono.\n";
```
Wymagania:
- Idempotentny: po updateOrderStatus zamowienie ma status_code='w_realizacji', wiec ponowne uruchomienie nie znajdzie nic.
- Bez recznego SQL — uzywaj OrdersRepository::updateOrderStatus (tworzy wpis w order_status_history + activity log).
- Bez parametrow CLI — operator uruchamia raz po deployu.
- Sprawdzic `bin/backfill_shipped_status_98.php` przed pisaniem — uzyc dokladnie tego samego patternu inicjalizacji (jak app jest dostepny: `require app.php` raz; jezeli phase 98 ma inny pattern, dopasowac).
- `php -l bin/backfill_payment_transition_111.php`
- DRY-RUN przez SELECT preview: `SELECT id, internal_order_number, source, status_code, payment_status FROM orders WHERE source IN ('allegro','shoppro') AND payment_status=2 AND LOWER(COALESCE(status_code,''))='nieoplacone';`
- Uruchomienie: `php bin/backfill_payment_transition_111.php` → log z liczba znalezionych + per-zamowienie [OK]
- Drugi run: `[backfill-111] Brak zamowien do naprawy.` (idempotencja)
- Spot-check w DB: `SELECT id, status_code FROM orders WHERE id IN (...)` → wszystkie `w_realizacji`
- `SELECT * FROM order_status_history WHERE order_id IN (...) ORDER BY id DESC LIMIT 5` — powinny byc wpisy `nieoplacone → w_realizacji` z change_source='import'
AC-3 spelnione: backfill naprawia historyczne dane idempotentnie.
Task 4: Aktualizacja dokumentacji codebase (architecture + tech_changelog)
.paul/codebase/architecture.md, .paul/codebase/tech_changelog.md
`.paul/codebase/architecture.md`:
- W sekcji "Order Lifecycle" rozszerz pkt 1 (Import) o: "Re-import: jezeli payment_status zmienia sie z 0/1 na 2, OrderImportRepository zwraca `payment_transition=true`, a service importu (AllegroOrderImportService, ShopproOrdersSyncService) emituje `payment.status_changed`. Chain regul automatyzacji (regula #7) wykonuje update_order_status → w_realizacji."
- Dodaj w "Key Decisions" (lub w odpowiedniej sekcji) krotki bullet o phase 111.
`.paul/codebase/tech_changelog.md`:
- Dodaj wpis na gore (chronologicznie): `## 2026-05-05 — Phase 111: Payment Transition Event` z opisem co i dlaczego (zamowienie #864 case).
PROJECT.md (`Validated (Shipped)` lista) — UNIFY phase doda wpis po zakonczeniu, NIE w PLAN/APPLY.
Manualny przeglad obu plikow — sekcje istnieja, formatowanie spojne z reszta.
Dokumentacja codebase zsynchronizowana z kodem.
DO NOT CHANGE
AllegroOrderImportService.php:99gateif ($wasCreated && ...)dlaorder.imported— Phase 98 decyzjaShopproPaymentStatusSyncService.php— istniejacy emitpayment.status_changedzostaje (idempotentnie wspolistnieje)- Regula automatyzacji #7 w DB — nie modyfikujemy warunkow ani akcji (akceptujemy ze obejmuje wszystkie integracje)
- Logika preservacji status_code w OrderImportRepository (Phase 62) —
$statusOverwriteAllowedmusi zachowac semantykecurrentStatus='nieoplacone' && newPaymentStatus===2 database/migrations/*— bez nowych migracji (idempotencja zalatwiona logika kodu, nie schema)
SCOPE LIMITS
- Bez zmian w
AutomationService::evaluateOrderStatusConditionani innych warunkach automatyzacji - Bez modyfikacji UI panelu automatyzacji
- Bez nowych pol w tabeli
orders(idempotencja przez logike, nie przez flage w DB) - Bez zmian w
AllegroStatusSyncService(korzysta z importSingleOrder, wiec dziedziczy fix automatycznie) - Bez testow jednostkowych dla nowej logiki (na uzytkownika decyzja czy dopisywac w UNIFY; tests/ nie ma obecnie pokrycia OrderImportRepository)
<success_criteria>
- Wszystkie 4 zadania ukonczone
- Wszystkie 4 AC zweryfikowane manualnie na lokalnej kopii DB lub na zamowieniu testowym
php -llint czysty- Brak nowych warningow PHP w logach po deploy </success_criteria>