- shopPRO: ShopproOrderMapper jako jedyne zrodlo heurystyki detekcji faktury; mapOrderAggregate() zwraca top-level invoice_detected (transient). - ShopproOrdersSyncService: usunieta wlasna shouldRequestInvoice(); propagacja aggregate['invoice_detected'] do setInvoiceRequested() tylko przy created=true. - Allegro: nowa shouldRequestInvoice(payload) z 4 wzorcami (invoice.required, naturalPerson=false, address.taxId, companyName/address.company.name). Wczesniej tylko invoice.required -> analogiczna luka jak shopPRO. - Migracja 20260513_000113: idempotentny backfill (UPDATE invoice_requested=1 WHERE is_invoice=1 AND invoice_requested=0) + DROP COLUMN orders.is_invoice. Guard przez information_schema.COLUMNS + PREPARE/EXECUTE z ALTER TABLE COMMENT no-op fallbackiem (portable MySQL/MariaDB). - Cleanup is_invoice z OrderImportRepository (INSERT cols/values/params, docstring Phase 112) i OrdersRepository (paginate SELECT, transformOrderRow hydrate). AllegroOrderImportService mapping w mapCheckoutFormPayload tez usuniety (wymuszone konsekwencja DROP COLUMN). - Bugfix #1089: zamowienie shopPRO z firm_nip (bez wants_invoice/invoice.required) ustawia teraz invoice_requested=1 -> UI w zakladce Platnosci zaznacza checkbox, przycisk "Wystaw fakture" widoczny. Pending operator: php bin/migrate.php (XAMPP MySQL online) -> backfill 7 zamowien. Smoke test: re-import shopPRO + nowe Allegro z NIP.
19 KiB
19 KiB
phase, plan, type, wave, depends_on, files_modified, autonomous, delegation
| phase | plan | type | wave | depends_on | files_modified | autonomous | delegation | |||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| 125-invoice-requested-import-fix | 01 | execute | 1 |
|
true | off |
Purpose
- Bugfix #1089: shopPRO order z
firm_nipnie ustawiainvoice_requested=1(operator musi ręcznie klikać toggle). - Analogiczna luka w Allegro: import nie wykrywa
invoice.address.taxIdbezinvoice.required=true. - Eliminacja struktury źródłowej buga: dwie kolumny dla tej samej semantyki (
is_invoiceustawiany przez importer,invoice_requestedczytany przez UI). Jedyne źródło prawdy →invoice_requested.
Output
- ShopproOrdersSyncService i AllegroOrderImportService używają tej samej heurystyki co mapper, propagując wynik z
$aggregate['order']['_invoice_detected'](lub bezpośrednio z mappera) dosetInvoiceRequested(). - Migracja idempotentna: backfill
invoice_requested=1dla zamówień gdzie poprzednio detekcja zadziałała na poziomie mappera, potem DROP COLUMNorders.is_invoice. - Kod oczyszczony z odniesień do
is_invoice(mapper, OrderImportRepository INSERT, OrdersRepository SELECT/hydrate). - Dokumentacja zaktualizowana (
db_schema.md,architecture.md,tech_changelog.md).
Stan w bazie produkcyjnej (2026-05-13):
- 7 zamówień
is_invoice=1 AND invoice_requested=0(w tym #1089, source=shoppro). - Kolumna
orders.is_invoice TINYINT(1) NOT NULL DEFAULT 0istnieje, nie ma indexu. db_schema.md(linia 244+) NIE wymienia kolumnyis_invoice— dokumentacja była już niespójna z DB.
Project Context
@.paul/PROJECT.md @.paul/ROADMAP.md @.paul/STATE.md
Prior Work
@.paul/phases/115-invoice-from-order/115-01-SUMMARY.md
Phase 115 wprowadziła dwutorową detekcję: mapper → is_invoice, sync service → invoice_requested.
Ta faza koryguje rozjazd przez propagację z mappera i eliminację is_invoice.
Source Files
@src/Modules/Settings/ShopproOrderMapper.php @src/Modules/Settings/ShopproOrdersSyncService.php @src/Modules/Settings/AllegroOrderImportService.php @src/Modules/Orders/OrderImportRepository.php @src/Modules/Orders/OrdersRepository.php @.paul/codebase/db_schema.md @.paul/codebase/architecture.md
<acceptance_criteria>
AC-1: Import shopPRO z firm_nip ustawia invoice_requested
Given zamówienie shopPRO z payloadem zawierającym `firm_name`+`firm_nip` (bez kluczy `wants_invoice`/`invoice_required`/`invoice.required`/`buyer.wants_invoice`/`buyer.invoice`)
When ShopproOrdersSyncService importuje to zamówienie jako nowe (wasCreated=true)
Then `orders.invoice_requested = 1`
And UI w zakładce Płatności (`/orders/{id}`) ma zaznaczony checkbox „Klient prosi o fakturę"
And przycisk „Wystaw fakturę" jest widoczny
AC-2: Import Allegro z NIP ustawia invoice_requested
Given zamówienie Allegro z payloadem `invoice.address.taxId` lub `invoice.naturalPerson=false`, ale BEZ `invoice.required=true`
When AllegroOrderImportService importuje to zamówienie jako nowe (wasCreated=true)
Then `orders.invoice_requested = 1`
AC-3: Re-import nie nadpisuje manualnego toggla
Given zamówienie z payload zawierającym firm_nip oraz manualnie ustawionym `invoice_requested=0` (operator odznaczył)
When importer re-importuje to zamówienie (wasCreated=false, delta-only)
Then `orders.invoice_requested` pozostaje `0`
And kontrakt Phase 115 (auto-set tylko przy `created=true`) jest zachowany
AC-4: Backfill istniejących zamówień
Given migracja `20260513_000113_drop_orders_is_invoice_and_backfill_invoice_requested.sql` zostaje uruchomiona
When `php bin/migrate.php` wykonuje ją na bazie gdzie 7 zamówień ma `is_invoice=1 AND invoice_requested=0`
Then `UPDATE orders SET invoice_requested=1 WHERE is_invoice=1 AND invoice_requested=0` wykonuje się PRZED `DROP COLUMN`
And po migracji `orders.is_invoice` nie istnieje
And dla każdego z 7 dotkniętych zamówień `invoice_requested=1`
And ponowne uruchomienie migracji jest no-op (kolumna już nie istnieje — `DROP COLUMN IF EXISTS` lub idempotentny SQL przez `information_schema` check)
AC-5: Kolumna is_invoice nie istnieje w runtime
Given migracja została wykonana
When aplikacja importuje nowe zamówienie lub wyświetla szczegóły zamówienia
Then żaden SELECT/INSERT/UPDATE nie referencuje `orders.is_invoice`
And testy PHPUnit (`OrderImportRepositoryTest`) przechodzą bez modyfikacji asercji powiązanych z is_invoice (albo asercje zostały usunięte wraz z fixture)
And ręczny `SELECT id, invoice_requested FROM orders WHERE id=1089` zwraca `invoice_requested=1`
</acceptance_criteria>
Task 1: Mapper exposes detection result, importers propagate it src/Modules/Settings/ShopproOrderMapper.php, src/Modules/Settings/ShopproOrdersSyncService.php, src/Modules/Settings/AllegroOrderImportService.php **ShopproOrderMapper.php:** - Usuń klucz `'is_invoice'` z tablicy `$order` (linia 148). - Zachowaj metodę `resolveInvoiceRequested(array $payload): bool` jako jedyne źródło heurystyki. - W `mapOrderAggregate()` dodaj do zwracanego array klucz top-level `'invoice_detected' => $this->resolveInvoiceRequested($payload)` (poza `order`, na poziomie aggregate'u — analogicznie do `addresses`/`items`/`payments`). Powód: nie zaśmiecać kontraktu `order` (który mapuje 1:1 na kolumny tabeli) flagą transient'ową.**ShopproOrdersSyncService.php:**
- W bloku po `upsertOrderAggregate` (okolice linii 273-277) zamień warunek `if ($this->shouldRequestInvoice($rawOrder))` na `if (!empty($aggregate['invoice_detected']))`.
- USUŃ prywatną metodę `shouldRequestInvoice(array $rawOrder): bool` (linie 316-338) — zastąpiona heurystyką mappera.
- Zachowaj guard `wasCreated=true` (kontrakt Phase 115 z AC-3).
**AllegroOrderImportService.php:**
- W bloku auto-set (linie 99-103) rozszerz warunek: `if (!empty($invoiceFlag['required']))` → wydziel do nowej prywatnej metody `private function shouldRequestInvoice(array $payload): bool` która sprawdza:
1. `!empty($payload['invoice']['required'])` (istniejące zachowanie)
2. `!empty($payload['invoice']['naturalPerson']) === false && isset($payload['invoice'])` → klient firmowy
3. `!empty($payload['invoice']['address']['taxId'])` — NIP w adresie faktury
4. `!empty($payload['invoice']['companyName'])`
- Zwracaj `true` gdy któryś z warunków spełniony.
- Wywołanie: `if ($wasCreated && $this->shouldRequestInvoice($payload)) { $this->ordersRepository->setInvoiceRequested($savedOrderId, true); }`
Avoid:
- NIE zmieniać sygnatury `OrdersRepository::setInvoiceRequested()` ani `recordActivity()`.
- NIE dodawać kolumny `invoice_detected` do DB — to wartość transient w pamięci między mapperem a syncem.
- `php -l src/Modules/Settings/ShopproOrderMapper.php` (lint pass).
- `php -l src/Modules/Settings/ShopproOrdersSyncService.php`.
- `php -l src/Modules/Settings/AllegroOrderImportService.php`.
- Grep: `grep -n "is_invoice" src/Modules/Settings/` zwraca tylko pozostałości w komentarzach (jeśli są) — żadnych aktywnych odwołań.
- Grep: `grep -n "shouldRequestInvoice" src/Modules/Settings/ShopproOrdersSyncService.php` → 0 trafień.
AC-1, AC-2, AC-3 satisfied: mapper jest jedynym źródłem detekcji shopPRO, Allegro ma rozszerzoną detekcję NIP/firmy, oba importery propagują flagę tylko przy created=true.
Task 2: Migration — backfill invoice_requested + DROP COLUMN is_invoice
database/migrations/20260513_000113_drop_orders_is_invoice_and_backfill_invoice_requested.sql
Utwórz migrację z idempotentnym DDL+DML:
```sql
-- Phase 125-01: backfill invoice_requested z is_invoice + DROP is_invoice
-- Idempotentna: DML w bloku IF (kolumna istnieje), DROP w bloku IF (kolumna istnieje).
-- Pattern z `.paul` Key Decision (2026-05-10): migracje no-op zawsze jako DDL.
SET @col_exists := (
SELECT COUNT(*) FROM information_schema.COLUMNS
WHERE TABLE_SCHEMA = DATABASE()
AND TABLE_NAME = 'orders'
AND COLUMN_NAME = 'is_invoice'
);
SET @sql_backfill := IF(@col_exists > 0,
'UPDATE orders SET invoice_requested = 1 WHERE is_invoice = 1 AND invoice_requested = 0',
'ALTER TABLE orders COMMENT = ''phase-125 backfill no-op'''
);
PREPARE stmt_backfill FROM @sql_backfill;
EXECUTE stmt_backfill;
DEALLOCATE PREPARE stmt_backfill;
SET @sql_drop := IF(@col_exists > 0,
'ALTER TABLE orders DROP COLUMN is_invoice',
'ALTER TABLE orders COMMENT = ''phase-125 drop no-op'''
);
PREPARE stmt_drop FROM @sql_drop;
EXECUTE stmt_drop;
DEALLOCATE PREPARE stmt_drop;
```
Avoid:
- NIE używać `SELECT 1;` jako no-op (Key Decision 2026-05-10: powoduje SQLSTATE 2014 z PDO unbuffered).
- NIE używać `DROP COLUMN IF EXISTS` (MariaDB only — produkcja może być na czystym MySQL).
- Plik istnieje w `database/migrations/`.
- Składnia SQL waliduje: `mysql --help` test dry-run niemożliwy bez DB; alternatywa: `mysql -h … -e "$(cat migration.sql)" --dry-run` — XAMPP nie wspiera; zostawiamy weryfikację na operatora przy `php bin/migrate.php`.
- Po wykonaniu na bazie: `SELECT COUNT(*) FROM orders WHERE invoice_requested=1;` rośnie o 7 vs. pre-migration baseline.
- `SHOW COLUMNS FROM orders LIKE 'is_invoice';` → empty set.
- Re-run migracji → no-op (ALTER TABLE COMMENT).
AC-4 satisfied: migracja idempotentna, backfill 7 zamówień przed DROP COLUMN, ponowne uruchomienie bezpieczne.
Task 3: Remove is_invoice from PHP code (repository, hydrate)
src/Modules/Orders/OrderImportRepository.php,
src/Modules/Orders/OrdersRepository.php
**OrderImportRepository.php:**
- W `insertOrder()` SQL (linie 161-175) usuń `is_invoice` z listy kolumn INSERT oraz `:is_invoice` z VALUES.
- W `orderParams()` (linia 258) usuń linię `'is_invoice' => !empty($orderData['is_invoice']) ? 1 : 0,`.
- W docstring `updateOrderDelta()` (linia 190) usuń wzmiankę o `is_invoice` z listy pól nie nadpisywanych przy delta-only (kolumna już nie istnieje, wzmianka jest dezinformująca).
**OrdersRepository.php:**
- Linia 172: usuń `o.is_invoice,` z SELECT (`findById()` / pokrewne — zweryfikować, że to ten sam zapytanie).
- Linia 235: usuń `'is_invoice' => (int) ($row['is_invoice'] ?? 0) === 1,` z hydrate / mapowania zwracanego row'a.
- Jeśli kontroler/view referencjuje `$order['is_invoice']` — sprawdź `grep -rn "\\['is_invoice'\\]" src/ resources/views/` i usuń odwołania (lub zamień na `$order['invoice_requested']` jeśli kontekst tego wymaga).
Avoid:
- NIE zmieniać kontraktu `OrderImportRepository::upsertOrderAggregate()` poza usunięciem klucza is_invoice (sygnatura, return value bez zmian).
- NIE dotykać `payment_status` ani `total_paid` (chronione przez Phase 119).
- `php -l src/Modules/Orders/OrderImportRepository.php`.
- `php -l src/Modules/Orders/OrdersRepository.php`.
- `grep -rn "is_invoice" src/ resources/views/` → 0 trafień (poza ewentualnymi historycznymi komentarzami w `.paul/`).
- Manual: po wdrożeniu importu zamówienie #1089 ma `invoice_requested=1` po re-imporcie (delta-only nie nadpisze, ale przed migracją można też ręcznie ustawić toggle aby zweryfikować że runtime nie próbuje pisać do is_invoice).
AC-5 satisfied: żaden runtime SELECT/INSERT nie odwołuje się do `is_invoice`; aplikacja działa po DROP COLUMN.
Task 4: Update docs (db_schema, architecture, tech_changelog)
.paul/codebase/db_schema.md,
.paul/codebase/architecture.md,
.paul/codebase/tech_changelog.md
**db_schema.md:**
- Sekcja `orders` (linia 244+) — pozostaje bez `is_invoice` (i tak już nieobecna, tylko potwierdzenie spójności).
- W komentarzu pod tabelą `orders` dodaj wzmiankę: „`invoice_requested` (Phase 113-01) jest jedynym znacznikiem żądania faktury. Legacy `is_invoice` usunięte w Phase 125-01."
- Phase footer: zaktualizuj `Updated: 2026-05-13` i `Total tables: 61` (bez zmian; kolumna usunięta to nie tabela).
**architecture.md:**
- W sekcji „### Auto-import flagi invoice_requested" (linia 261+) zaktualizuj opis:
- shopPRO: `ShopproOrderMapper::resolveInvoiceRequested()` jest jedynym źródłem heurystyki; sprawdza: `is_invoice` w payload (top-level z legacy źródeł zewnętrznych — payload, nie kolumna), `invoice.required`, top-level `invoice` jako bool, `firm_name`/`firm_nip`/`invoice.company_name`/`invoice.tax_id`. Sync service propaguje wynik z `$aggregate['invoice_detected']`.
- Allegro: `AllegroOrderImportService::shouldRequestInvoice($payload)` sprawdza `invoice.required`, `invoice.naturalPerson=false`, `invoice.address.taxId`, `invoice.companyName`. Tylko przy `wasCreated=true`.
**tech_changelog.md:**
- Dodaj wpis dla Phase 125-01:
```
## Phase 125-01 — invoice_requested import fix (2026-05-13)
- Bug: zamówienia shopPRO z `firm_nip` (bez `invoice.required`) nie ustawiały `invoice_requested=1` (mapper wykrywał, sync service nie).
- Fix: ShopproOrderMapper eksponuje `invoice_detected` w aggregate. ShopproOrdersSyncService propaguje zamiast duplikować heurystykę. Usunięta metoda `shouldRequestInvoice` ze sync service.
- Allegro: rozszerzenie detekcji o `invoice.naturalPerson=false`, `invoice.address.taxId`, `invoice.companyName`.
- Migracja 20260513_000113 — backfill 7 zamówień + DROP COLUMN `orders.is_invoice` (legacy z Phase 115, dryft względem `invoice_requested`).
- Files: ShopproOrderMapper.php, ShopproOrdersSyncService.php, AllegroOrderImportService.php, OrderImportRepository.php, OrdersRepository.php.
```
- Grep `grep -n "is_invoice" .paul/codebase/db_schema.md .paul/codebase/architecture.md` — tylko historyczne wzmianki w kontekście Phase 125.
- Sekcja Phase 125 istnieje w `tech_changelog.md`.
Dokumentacja spójna z kodem; przyszli czytelnicy widzą że `invoice_requested` to jedyne źródło prawdy.
DO NOT CHANGE
orders.invoice_requestedschema (Phase 113-01 zostaje, indexidx_orders_invoice_requestedzostaje).OrdersRepository::setInvoiceRequested()sygnatura irecordActivity('invoice_requested_changed')event type.- Kontrakt Phase 115: auto-set tylko przy
wasCreated=true, delta-only re-import nie nadpisuje manualnej flagi. - Kontrakt Phase 119:
updateOrderDelta()zachowuje ochronętotal_paidiis_canceled_by_buyer— ta faza nie dotyka logiki delta. - Kontrakt Phase 112:
replaceAddresses/replaceItems/replaceNoteswywoływane tylko przycreated=true. payload_json— nie filtrujemy, surowy payload zostaje (mapper-tylko detekcja).- Allegro
invoice.requiredjako warunek wystarczający — pozostaje (rozszerzenie, nie zamiana).
SCOPE LIMITS
- NIE robimy nowego eventu automatyzacji
invoice.created(zostaje na future plan zgodnie z Key Decision 2026-05-10). - NIE dotykamy INVOICE-IDEMP-115 (double-POST Fakturownia) — to osobny todo.
- NIE refaktorujemy
OrderImportRepository::orderParams()poza usunięciem klucza is_invoice (tylko minimalna delta). - NIE migrujemy stałych Allegro do wspólnego helpera —
shouldRequestInvoicezostaje prywatną metodą per importer (shopPRO ma to w mapperze, Allegro nie ma analogicznego mappera klasowego; ujednolicenie poza zakresem). - NIE dodajemy testów PHPUnit dla detekcji shopPRO/Allegro (możliwe w follow-up; weryfikacja przez ręczny smoke test operatora na zamówieniu #1089).
<success_criteria>
- 5 plików PHP zmodyfikowanych zgodnie z zadaniami 1+3.
- Nowa migracja idempotentna.
- 3 pliki dokumentacji zaktualizowane.
- AC-1..AC-5 spełnione (manualna weryfikacja AC-1/AC-4 przez operatora na żywej bazie po uruchomieniu migracji).
- Order #1089 po backfillu ma
invoice_requested=1i checkbox w UI zaznaczony. </success_criteria>