Files
orderPRO/.paul/phases/125-invoice-requested-import-fix/125-01-PLAN.md
Jacek Pyziak 2ab461aaae feat(125): invoice_requested import fix + drop legacy is_invoice column
- 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.
2026-05-12 22:11:49 +02:00

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
src/Modules/Settings/ShopproOrdersSyncService.php
src/Modules/Settings/ShopproOrderMapper.php
src/Modules/Settings/AllegroOrderImportService.php
src/Modules/Orders/OrderImportRepository.php
src/Modules/Orders/OrdersRepository.php
database/migrations/20260513_000113_drop_orders_is_invoice_and_backfill_invoice_requested.sql
.paul/codebase/db_schema.md
.paul/codebase/architecture.md
.paul/codebase/tech_changelog.md
true off
## Goal Naprawić rozjazd między mapperem shopPRO a auto-set `orders.invoice_requested` przy imporcie, tak aby zamówienie z fakturą poprawnie pokazywało zaznaczony checkbox w zakładce Płatności i odblokowywało przycisk „Wystaw fakturę". Przy okazji usunąć duplikującą się kolumnę `orders.is_invoice` (legacy z Phase 115), bo to ona była źródłem dryftu między mapperem a sync service'em.

Purpose

  • Bugfix #1089: shopPRO order z firm_nip nie ustawia invoice_requested=1 (operator musi ręcznie klikać toggle).
  • Analogiczna luka w Allegro: import nie wykrywa invoice.address.taxId bez invoice.required=true.
  • Eliminacja struktury źródłowej buga: dwie kolumny dla tej samej semantyki (is_invoice ustawiany przez importer, invoice_requested czytany 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) do setInvoiceRequested().
  • Migracja idempotentna: backfill invoice_requested=1 dla zamówień gdzie poprzednio detekcja zadziałała na poziomie mappera, potem DROP COLUMN orders.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).
- **[Detekcja shopPRO]** — Jak szeroko rozszerzyć auto-detekcję faktury przy imporcie shopPRO? → Odpowiedź: Propaguj wynik mappera (is_invoice → invoice_requested) — najprościej, zero duplikacji, mapper już sprawdza firm_name/firm_nip/invoice.required/is_invoice/invoice (top-level). - **[Allegro fix]** — Czy przy okazji naprawić analogiczną lukę dla Allegro? → Odpowiedź: Tak — rozszerz Allegro o detekcję NIP/firmy (`invoice.address.taxId`, `invoice.naturalPerson=false`, `invoice.companyName`). - **[Backfill]** — Backfill istniejących zamówień: jak wykonać? → Odpowiedź: Migracja SQL idempotentna (UPDATE … WHERE is_invoice=1 AND invoice_requested=0) — pipeline `php bin/migrate.php`, idempotentne (po backfillu DROP COLUMN i tak czyni operację jednorazową). - **[is_invoice fate]** — Co zrobić z kolumną orders.is_invoice? → Odpowiedź: Usunąć kolumnę w tej fazie — migracja DROP COLUMN po backfillu, usunięcie z `OrderImportRepository` (INSERT, params, docstring Phase 112), z `OrdersRepository` (SELECT, hydrate), z `ShopproOrderMapper` (zwraca flagę jako transient w aggregate'cie zamiast pisać do DB), z `AllegroOrderImportService` (analogicznie).

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 0 istnieje, nie ma indexu.
  • db_schema.md (linia 244+) NIE wymienia kolumny is_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_requested schema (Phase 113-01 zostaje, index idx_orders_invoice_requested zostaje).
  • OrdersRepository::setInvoiceRequested() sygnatura i recordActivity('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_paid i is_canceled_by_buyer — ta faza nie dotyka logiki delta.
  • Kontrakt Phase 112: replaceAddresses/replaceItems/replaceNotes wywoływane tylko przy created=true.
  • payload_json — nie filtrujemy, surowy payload zostaje (mapper-tylko detekcja).
  • Allegro invoice.required jako 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 — shouldRequestInvoice zostaje 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).
Przed zamknięciem planu: - [ ] `php -l` na 5 zmodyfikowanych plikach PHP — bez błędów składni. - [ ] `grep -rn "is_invoice" src/` zwraca 0 trafień w aktywnym kodzie. - [ ] Migracja przygotowana, idempotentna (`@col_exists` guard + ALTER TABLE COMMENT no-op). - [ ] Dokumentacja zaktualizowana (db_schema, architecture, tech_changelog). - [ ] STATE.md pokazuje pending action: operator uruchamia `php bin/migrate.php` + ręczna weryfikacja na #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=1 i checkbox w UI zaznaczony. </success_criteria>
Po zakończeniu utworzyć `.paul/phases/125-invoice-requested-import-fix/125-01-SUMMARY.md`.