Phase 29 complete (v1.3): - Tabela delivery_status_mappings z DB overrides - DeliveryStatus: normalizeWithOverrides(), descriptionWithOverrides(), getDefaultMappings() - UI ustawień: tabela mapowań per provider (InPost/Apaczka/Allegro), bulk save, reset, resetAll - 5 endpointów w routes/web.php, link w menu bocznym Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
11 KiB
11 KiB
phase, plan, type, wave, depends_on, files_modified, autonomous
| phase | plan | type | wave | depends_on | files_modified | autonomous | |||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| 29-delivery-status-mapping-ui | 01 | execute | 1 |
|
|
false |
Purpose
Każdy przewoźnik zwraca inne statusy (InPost: adopted_at_sorting_center, Apaczka: NEW/CONFIRMED, Allegro: IN_TRANSIT). Obecnie mapowania są zahardkodowane w DeliveryStatus.php. Użytkownik powinien móc:
- Zobaczyć wszystkie mapowania (domyślne + własne)
- Zmienić przypisanie surowego statusu do innego znormalizowanego statusu
- Zmienić opis wyświetlany przy surowym statusie
Output
- Tabela DB
delivery_status_mappingsna custom overrides - Strona ustawień z tabelą mapowań per provider
- DeliveryStatus sprawdza DB overrides przed fallback na stałe
Prior Work
@.paul/phases/28-shipment-tracking-ui/28-01-SUMMARY.md
Source Files
@src/Modules/Shipments/DeliveryStatus.php @src/Modules/Settings/AllegroStatusMappingController.php @src/Modules/Settings/AllegroStatusMappingRepository.php @resources/views/settings/statuses.php @routes/web.php
## Required Skills (from SPECIAL-FLOWS.md)| Skill | Priority | When to Invoke | Loaded? |
|---|---|---|---|
| sonar-scanner | required | Po APPLY, przed UNIFY | ○ |
Skill Invocation Checklist
- sonar-scanner uruchomiony po zakończeniu APPLY
<acceptance_criteria>
AC-1: Lista mapowań per provider
Given użytkownik jest na stronie Ustawienia > Mapowanie statusów dostawy
When wybiera providera (InPost / Apaczka / Allegro)
Then widzi tabelę z kolumnami: Status surowy, Opis, Status znormalizowany
And każdy wiersz pokazuje aktualne przypisanie (domyślne lub custom)
And wiersze z custom override są wyróżnione wizualnie
AC-2: Edycja mapowania statusu
Given użytkownik widzi tabelę mapowań dla wybranego providera
When zmienia "Status znormalizowany" w wierszu (select z opcjami: Utworzona, Potwierdzona, W tranzycie, etc.)
And klika Zapisz
Then mapowanie jest zapisane w tabeli delivery_status_mappings
And przy następnym pobraniu statusu z API, nowe mapowanie jest używane
AC-3: Edycja opisu statusu
Given użytkownik widzi tabelę mapowań
When zmienia tekst w kolumnie "Opis" dla surowego statusu
And klika Zapisz
Then opis jest zapisany w delivery_status_mappings
And tooltip w badge'u statusu pokazuje nowy opis
AC-4: Reset do domyślnych
Given użytkownik ma custom override dla statusu
When klika przycisk "Resetuj" obok wiersza
Then custom override jest usunięty z delivery_status_mappings
And mapowanie wraca do domyślnego z DeliveryStatus.php
AC-5: DeliveryStatus używa custom overrides
Given istnieje wpis w delivery_status_mappings (provider='apaczka', raw_status='NEW', normalized_status='confirmed')
When system wywołuje DeliveryStatus::normalize('apaczka', 'NEW')
Then zwraca 'confirmed' (z DB) zamiast 'created' (domyślny)
</acceptance_criteria>
Task 1: Migracja DB + Repository database/migrations/20260323_000070_create_delivery_status_mappings_table.sql, src/Modules/Shipments/DeliveryStatusMappingRepository.php **Migracja — tabela delivery_status_mappings:** ```sql CREATE TABLE IF NOT EXISTS delivery_status_mappings ( id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY, provider VARCHAR(32) NOT NULL, raw_status VARCHAR(64) NOT NULL, normalized_status VARCHAR(32) NOT NULL, description VARCHAR(255) NOT NULL DEFAULT '', created_at DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, updated_at DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, UNIQUE KEY uq_provider_raw (provider, raw_status) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci; ``` Idempotentna (IF NOT EXISTS).**DeliveryStatusMappingRepository:**
- `listByProvider(string $provider): array` — SELECT * WHERE provider = :provider
- `upsertMapping(string $provider, string $rawStatus, string $normalizedStatus, string $description): void` — INSERT ON DUPLICATE KEY UPDATE
- `deleteMapping(string $provider, string $rawStatus): void` — DELETE WHERE provider AND raw_status
- `getAllOverrides(): array` — SELECT * (dla cache w DeliveryStatus)
Wzoruj na AllegroStatusMappingRepository (konstruktor z PDO, prepared statements).
Avoid: nie dodawaj logiki biznesowej do repozytorium.
php -l na obu plikach
Migracja wykonuje się bez błędów na DB
Tabela DB gotowa, repozytorium CRUD działa — fundament dla AC-2, AC-3, AC-4
Task 2: DeliveryStatus — DB overrides + kontroler + widok + routing
src/Modules/Shipments/DeliveryStatus.php, src/Modules/Settings/DeliveryStatusMappingController.php, resources/views/settings/delivery-status-mappings.php, routes/web.php
**DeliveryStatus — metody z DB override:**
- Dodaj nowe statyczne metody:
- `normalizeWithOverrides(string $provider, string $rawStatus, array $overrides): string`
- `descriptionWithOverrides(string $provider, string $rawStatus, array $overrides): string`
- $overrides to tablica ['provider:raw_status' => ['normalized_status' => X, 'description' => Y]]
- Najpierw szukaj w $overrides, potem fallback na istniejące stałe (INPOST_MAP, etc.)
- Dodaj `getDefaultMappings(string $provider): array` — zwraca wszystkie mapowania z hardkodowanych stałych jako tablicę [raw_status => [normalized, description]]
**DeliveryStatusMappingController:**
- Konstruktor: Template, Translator, AuthService, DeliveryStatusMappingRepository
- `index(Request): Response`:
- Parametr GET `provider` (domyślnie 'inpost')
- Pobierz domyślne mapowania z DeliveryStatus::getDefaultMappings($provider)
- Pobierz custom overrides z repo: listByProvider($provider)
- Merge: custom nadpisuje domyślne
- Przekaż do widoku: provider, mappings, providers list, csrfToken, normalized status options
- `save(Request): Response`:
- Waliduj CSRF
- Odczytaj POST: provider, raw_status, normalized_status, description
- Waliduj: normalized_status musi być jednym z DeliveryStatus::ALL_STATUSES
- Wywołaj repo->upsertMapping(...)
- Flash success, redirect
- `reset(Request): Response`:
- Waliduj CSRF
- Odczytaj POST: provider, raw_status
- Wywołaj repo->deleteMapping(...)
- Flash success, redirect
**Widok delivery-status-mappings.php:**
- Nawigacja providerów: InPost | Apaczka | Allegro (linki z ?provider=X)
- Tabela z kolumnami:
- Status surowy (readonly, tekst)
- Opis (input text, edytowalny)
- Status znormalizowany (select z opcjami z LABEL_PL)
- Akcje: Zapisz (jeśli zmieniony) | Resetuj (jeśli custom)
- Każdy wiersz jako osobny mini-formularz POST (jak w statuses.php pattern)
- Wiersze z custom override: dodaj klasę CSS `is-custom` (np. lekkie tło)
- Formularz bulk save: jeden przycisk "Zapisz wszystkie" z JS zbierającym dane
**routes/web.php:**
- GET `/settings/delivery-status-mappings` → index
- POST `/settings/delivery-status-mappings/save` → save
- POST `/settings/delivery-status-mappings/reset` → reset
- Wszystkie z $authMiddleware
- Zainicjalizuj kontroler w sekcji controller instantiation
**Menu nawigacji:**
- Dodaj link "Mapowanie statusów" w menu ustawień (layout/sidebar)
Avoid:
- NIE zmieniaj istniejących metod normalize() i description() — nowe metody obok
- NIE usuwaj hardkodowanych stałych — to fallback
- NIE dodawaj JS frameworków — czyste formularze HTML
php -l na wszystkich zmienionych plikach PHP
Strona /settings/delivery-status-mappings ładuje się bez błędów
Zmiana mapowania i zapis działa
AC-1, AC-2, AC-3, AC-4, AC-5 satisfied
Strona ustawień mapowania statusów dostawy — tabela mapowań per provider, edycja przypisań i opisów, reset do domyślnych.
1. Otwórz Ustawienia > Mapowanie statusów dostawy
2. Sprawdź zakładkę InPost — tabela ze statusami i ich opisami
3. Przełącz na Apaczka — sprawdź czy statusy tekstowe (NEW, CONFIRMED) są widoczne
4. Zmień mapowanie jednego statusu (np. NEW → Potwierdzona zamiast Utworzona) i Zapisz
5. Sprawdź czy badge w zamówieniu #22 odzwierciedla zmianę
6. Kliknij Resetuj — sprawdź czy wraca do domyślnego
7. Zmień opis statusu i sprawdź tooltip w badge'u
Type "approved" to continue, or describe issues to fix
DO NOT CHANGE
- src/Modules/Shipments/ShipmentTrackingInterface.php
- src/Modules/Shipments/InpostTrackingService.php
- src/Modules/Shipments/ApaczkaTrackingService.php
- src/Modules/Shipments/AllegroTrackingService.php
- src/Modules/Cron/ShipmentTrackingHandler.php
- Istniejące stałe w DeliveryStatus.php (INPOST_MAP, APACZKA_MAP, etc.) — to fallback
SCOPE LIMITS
- Tylko UI do konfiguracji mapowań — bez zmian w logice trackingu
- Brak nowych zależności npm/composer
- Brak JavaScript frameworków — czyste formularze HTML + POST
- Tracking services (InPost/Apaczka/Allegro) muszą przekazywać overrides do DeliveryStatus — to integracja w ShipmentTrackingHandler, nie w tym planie jeśli wymaga dużych zmian
<success_criteria>
- Tabela delivery_status_mappings utworzona
- Strona ustawień działa z 3 providerami
- Edycja mapowania i opisu zapisuje się poprawnie
- Reset do domyślnych działa
- DeliveryStatus respektuje custom overrides
- Brak nowych zależności </success_criteria>