fix: scontainers edit saves existing record instead of creating new
Fixes static container admin edit flow by preserving id in hiddenFields and adding route-id fallback during save. Adds regression tests for edit/create id behavior, updates release docs (changelog/testing/CLAUDE), and appends SonarQube open issues to docs/TODO. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -1,12 +1,12 @@
|
|||||||
# shopPRO
|
# shopPRO
|
||||||
|
|
||||||
## What This Is
|
## What This Is
|
||||||
|
|
||||||
Autorski silnik sklepu internetowego pisany od podstaw — odpowiednik WooCommerce lub PrestaShop, ale bez zależności od zewnętrznych platform. Składa się z panelu administratora (zarządzanie zamówieniami, produktami, klientami) oraz części frontowej dla klienta końcowego.
|
Autorski silnik sklepu internetowego pisany od podstaw — odpowiednik WooCommerce lub PrestaShop, ale bez zależności od zewnętrznych platform. Składa się z panelu administratora (zarządzanie zamówieniami, produktami, klientami) oraz części frontowej dla klienta końcowego.
|
||||||
|
|
||||||
## Core Value
|
## Core Value
|
||||||
|
|
||||||
Właściciel sklepu internetowego ma pełną kontrolę nad sprzedażą online — produktami, zamówieniami i klientami — w jednym spójnym systemie pisanym od podstaw, bez narzutów zewnętrznych platform.
|
Właściciel sklepu internetowego ma pełną kontrolę nad sprzedażą online — produktami, zamówieniami i klientami — w jednym spójnym systemie pisanym od podstaw, bez narzutów zewnętrznych platform.
|
||||||
|
|
||||||
## Current State
|
## Current State
|
||||||
|
|
||||||
@@ -14,22 +14,22 @@ Właściciel sklepu internetowego ma pełną kontrolę nad sprzedażą online
|
|||||||
|-----------|-------|
|
|-----------|-------|
|
||||||
| Version | 0.333 |
|
| Version | 0.333 |
|
||||||
| Status | Production |
|
| Status | Production |
|
||||||
| Last Updated | 2026-03-12 |
|
| Last Updated | 2026-04-18 |
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
### Validated (Shipped)
|
### Validated (Shipped)
|
||||||
|
|
||||||
- [x] Panel administratora — zarządzanie produktami, kategoriami, atrybutami
|
- [x] Panel administratora — zarządzanie produktami, kategoriami, atrybutami
|
||||||
- [x] Panel administratora — zarządzanie zamówieniami
|
- [x] Panel administratora — zarządzanie zamówieniami
|
||||||
- [x] Panel administratora — zarządzanie klientami
|
- [x] Panel administratora — zarządzanie klientami
|
||||||
- [x] Część frontowa — przeglądanie i kupowanie produktów
|
- [x] Część frontowa — przeglądanie i kupowanie produktów
|
||||||
- [x] Koszyk i składanie zamówień
|
- [x] Koszyk i składanie zamówień
|
||||||
- [x] Integracje płatności i dostaw
|
- [x] Integracje płatności i dostaw
|
||||||
- [x] REST API (ordersPRO + Ekomi)
|
- [x] REST API (ordersPRO + Ekomi)
|
||||||
- [x] Redis caching
|
- [x] Redis caching
|
||||||
- [x] Ochrona przed podwójnym składaniem zamówienia
|
- [x] Ochrona przed podwójnym składaniem zamówienia
|
||||||
- [x] Domain-Driven Architecture (migracja z legacy zakończona)
|
- [x] Domain-Driven Architecture (migracja z legacy zakończona)
|
||||||
|
|
||||||
### Active (In Progress)
|
### Active (In Progress)
|
||||||
|
|
||||||
@@ -41,16 +41,16 @@ Właściciel sklepu internetowego ma pełną kontrolę nad sprzedażą online
|
|||||||
|
|
||||||
### Out of Scope
|
### Out of Scope
|
||||||
|
|
||||||
- Multitenancy (wiele sklepów w jednej instancji) — nie planowane
|
- Multitenancy (wiele sklepów w jednej instancji) — nie planowane
|
||||||
|
|
||||||
## Target Users
|
## Target Users
|
||||||
|
|
||||||
**Primary:** Właściciel/administrator sklepu internetowego
|
**Primary:** Właściciel/administrator sklepu internetowego
|
||||||
- Zarządza produktami, zamówieniami, klientami przez panel admina
|
- ZarzÄ…dza produktami, zamĂłwieniami, klientami przez panel admina
|
||||||
- Potrzebuje niezawodnego, szybkiego narzędzia bez zbędnych zależności
|
- Potrzebuje niezawodnego, szybkiego narzędzia bez zbędnych zależności
|
||||||
|
|
||||||
**Secondary:** Klient końcowy sklepu
|
**Secondary:** Klient końcowy sklepu
|
||||||
- Przegląda produkty, dodaje do koszyka, składa zamówienia
|
- Przegląda produkty, dodaje do koszyka, składa zamówienia
|
||||||
|
|
||||||
## Context
|
## Context
|
||||||
|
|
||||||
@@ -58,26 +58,27 @@ Właściciel sklepu internetowego ma pełną kontrolę nad sprzedażą online
|
|||||||
- PHP 7.4+ (produkcja: PHP < 8.0)
|
- PHP 7.4+ (produkcja: PHP < 8.0)
|
||||||
- Medoo ORM (`$mdb`), Redis caching
|
- Medoo ORM (`$mdb`), Redis caching
|
||||||
- Domain-Driven Design z Dependency Injection
|
- Domain-Driven Design z Dependency Injection
|
||||||
- PHPUnit 9.6, 810+ testów
|
- PHPUnit 9.6, 810+ testĂłw
|
||||||
- Namespace: `\Domain\`, `\admin\`, `\front\`, `\api\`, `\Shared\`
|
- Namespace: `\Domain\`, `\admin\`, `\front\`, `\api\`, `\Shared\`
|
||||||
|
|
||||||
## Constraints
|
## Constraints
|
||||||
|
|
||||||
### Technical Constraints
|
### Technical Constraints
|
||||||
- PHP < 8.0 na produkcji (brak `match`, named arguments, union types)
|
- PHP < 8.0 na produkcji (brak `match`, named arguments, union types)
|
||||||
- Medoo ORM — prepared statements bez wyjątków
|
- Medoo ORM — prepared statements bez wyjątków
|
||||||
- Redis wymagany dla cache
|
- Redis wymagany dla cache
|
||||||
|
|
||||||
### Business Constraints
|
### Business Constraints
|
||||||
- System wdrażany u klientów jako update package (ZIP)
|
- System wdraĹĽany u klientĂłw jako update package (ZIP)
|
||||||
|
|
||||||
## Key Decisions
|
## Key Decisions
|
||||||
|
|
||||||
| Decision | Rationale | Date | Status |
|
| Decision | Rationale | Date | Status |
|
||||||
|----------|-----------|------|--------|
|
|----------|-----------|------|--------|
|
||||||
| DDD + DI zamiast legacy architektury | Testowalność, separacja odpowiedzialności | 2025 | Active |
|
| DDD + DI zamiast legacy architektury | Testowalność, separacja odpowiedzialności | 2025 | Active |
|
||||||
| PHP < 8.0 kompatybilność | Klienci na starszych serwerach | 2025 | Active |
|
| PHP < 8.0 kompatybilność | Klienci na starszych serwerach | 2025 | Active |
|
||||||
| Własny silnik zamiast frameworka | Pełna kontrola, brak narzutów | - | Active |
|
| Własny silnik zamiast frameworka | Pełna kontrola, brak narzutów | - | Active |
|
||||||
|
| `id` w tabbed FormEdit przez `hiddenFields` | Zapobiega insert zamiast update przy edycji encji | 2026-04-18 | Active |
|
||||||
|
|
||||||
## Success Metrics
|
## Success Metrics
|
||||||
|
|
||||||
@@ -93,7 +94,7 @@ Właściciel sklepu internetowego ma pełną kontrolę nad sprzedażą online
|
|||||||
| Backend | PHP 7.4+ | < 8.0 na produkcji |
|
| Backend | PHP 7.4+ | < 8.0 na produkcji |
|
||||||
| ORM | Medoo | `$mdb` global |
|
| ORM | Medoo | `$mdb` global |
|
||||||
| Cache | Redis | CacheHandler singleton |
|
| Cache | Redis | CacheHandler singleton |
|
||||||
| Frontend | HTML/CSS/JS | Własny silnik szablonów (Tpl) |
|
| Frontend | HTML/CSS/JS | Własny silnik szablonów (Tpl) |
|
||||||
| Auth | Sesje PHP | CSRF, XSS protection |
|
| Auth | Sesje PHP | CSRF, XSS protection |
|
||||||
| Testy | PHPUnit 9.6 | phpunit.phar |
|
| Testy | PHPUnit 9.6 | phpunit.phar |
|
||||||
|
|
||||||
@@ -102,14 +103,15 @@ Właściciel sklepu internetowego ma pełną kontrolę nad sprzedażą online
|
|||||||
See: .paul/SPECIAL-FLOWS.md
|
See: .paul/SPECIAL-FLOWS.md
|
||||||
|
|
||||||
Quick Reference:
|
Quick Reference:
|
||||||
- /feature-dev → Nowe funkcje, większe zmiany (required)
|
- /feature-dev → Nowe funkcje, większe zmiany (required)
|
||||||
- /koniec-pracy → Release, update package (required)
|
- /koniec-pracy → Release, update package (required)
|
||||||
- /frontend-design → Komponenty UI, szablony widoków
|
- /frontend-design → Komponenty UI, szablony widoków
|
||||||
- /code-review → Przegląd kodu przed release
|
- /code-review → Przegląd kodu przed release
|
||||||
- /simplify → Upraszczanie po implementacji
|
- /simplify → Upraszczanie po implementacji
|
||||||
- /claude-md-improver → Utrzymanie CLAUDE.md
|
- /claude-md-improver → Utrzymanie CLAUDE.md
|
||||||
- /zapisz + /wznow → Zapis i wznowienie sesji
|
- /zapisz + /wznow → Zapis i wznowienie sesji
|
||||||
|
|
||||||
---
|
---
|
||||||
*PROJECT.md — Updated when requirements or context change*
|
*PROJECT.md — Updated when requirements or context change*
|
||||||
*Last updated: 2026-03-12*
|
*Last updated: 2026-04-18 after Phase 15*
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
# Roadmap: shopPRO
|
# Roadmap: shopPRO
|
||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
shopPRO to autorski silnik sklepu internetowego rozwijany iteracyjnie. Projekt jest już na produkcji (v0.333) — roadmap obejmuje planowane funkcje i usprawnienia kolejnych wersji.
|
shopPRO to autorski silnik sklepu internetowego rozwijany iteracyjnie. Projekt jest już na produkcji (v0.333) — roadmap obejmuje planowane funkcje i usprawnienia kolejnych wersji.
|
||||||
|
|
||||||
## Current Milestone
|
## Current Milestone
|
||||||
|
|
||||||
**Security hardening** (v0.33x)
|
**Hotfix backlog**
|
||||||
Status: In progress
|
Status: Complete
|
||||||
Phases: 3 of 4 complete
|
Phases: 4 of 4 complete
|
||||||
|
|
||||||
## Phases
|
## Phases
|
||||||
|
|
||||||
@@ -16,26 +16,27 @@ Phases: 3 of 4 complete
|
|||||||
|-------|------|-------|--------|-----------|
|
|-------|------|-------|--------|-----------|
|
||||||
| 1 | Sensitive data logging fix | 1 | Done | 2026-03 |
|
| 1 | Sensitive data logging fix | 1 | Done | 2026-03 |
|
||||||
| 2 | Path traversal + XSS escaping | 1 | Done | 2026-03 (v0.335) |
|
| 2 | Path traversal + XSS escaping | 1 | Done | 2026-03 (v0.335) |
|
||||||
| 3 | Error handling w krytycznych ścieżkach | 1 | Done | 2026-03 (v0.336) |
|
| 3 | Error handling w krytycznych ścieżkach | 1 | Done | 2026-03 (v0.336) |
|
||||||
| 4 | CSRF protection — admin panel forms | 1 | Applied | 2026-03 (v0.337) |
|
| 4 | CSRF protection — admin panel forms | 1 | Applied | 2026-03 (v0.337) |
|
||||||
| 5 | Order bugs fix — duplicate + COD status | 1 | Applied | 2026-03 (v0.338) |
|
| 5 | Order bugs fix — duplicate + COD status | 1 | Applied | 2026-03 (v0.338) |
|
||||||
|
|
||||||
## Next Milestone
|
## Next Milestone
|
||||||
|
|
||||||
**Tech debt — Integrations refactoring**
|
**Tech debt — Integrations refactoring**
|
||||||
Status: Planning
|
Status: Planning
|
||||||
|
|
||||||
| Phase | Name | Plans | Status | Completed |
|
| Phase | Name | Plans | Status | Completed |
|
||||||
|-------|------|-------|--------|-----------|
|
|-------|------|-------|--------|-----------|
|
||||||
| 6 | IntegrationsRepository split → ApiloRepository | 2 | Done | 2026-03 |
|
| 6 | IntegrationsRepository split → ApiloRepository | 2 | Done | 2026-03 |
|
||||||
|
|
||||||
## Hotfix
|
## Hotfix
|
||||||
|
|
||||||
| Phase | Name | Plans | Status | Completed |
|
| Phase | Name | Plans | Status | Completed |
|
||||||
|-------|------|-------|--------|-----------|
|
|-------|------|-------|--------|-----------|
|
||||||
| 7 | Coupon Fatal Error — order placement crash | 1 | Done | 2026-03-15 |
|
| 7 | Coupon Fatal Error — order placement crash | 1 | Done | 2026-03-15 |
|
||||||
| 8 | Apilo orders not sending — diagnoza i naprawa | 1 | Done | 2026-03-16 |
|
| 8 | Apilo orders not sending — diagnoza i naprawa | 1 | Done | 2026-03-16 |
|
||||||
| 9 | Apilo email notification + infinite retry | 1 | Done | 2026-03-19 |
|
| 9 | Apilo email notification + infinite retry | 1 | Done | 2026-03-19 |
|
||||||
|
| 15 | Scontainers edit saves as new record | 1 | Done | 2026-04-18 |
|
||||||
|
|
||||||
## Feature
|
## Feature
|
||||||
|
|
||||||
@@ -43,65 +44,72 @@ Status: Planning
|
|||||||
|-------|------|-------|--------|-----------|
|
|-------|------|-------|--------|-----------|
|
||||||
| 10 | Edycja personalizacji produktu w koszyku | 1 | Done | 2026-03-19 |
|
| 10 | Edycja personalizacji produktu w koszyku | 1 | Done | 2026-03-19 |
|
||||||
| 11 | DataLayer GA4 analytics fix | 1 | Done | 2026-03-25 |
|
| 11 | DataLayer GA4 analytics fix | 1 | Done | 2026-03-25 |
|
||||||
| 12 | summaryView redirect fix — double order block | 1 | Done | 2026-03-25 |
|
| 12 | summaryView redirect fix — double order block | 1 | Done | 2026-03-25 |
|
||||||
| 13 | Basket logging + TTL token fix | 1 | Done | 2026-03-25 |
|
| 13 | Basket logging + TTL token fix | 1 | Done | 2026-03-25 |
|
||||||
| 14 | Custom fields delete bug — usunięcie wszystkich pól | 1 | Done | 2026-04-16 |
|
| 14 | Custom fields delete bug — usunięcie wszystkich pól | 1 | Done | 2026-04-16 |
|
||||||
|
|
||||||
## Phase Details
|
## Phase Details
|
||||||
|
|
||||||
### Phase 4 — CSRF protection
|
### Phase 4 — CSRF protection
|
||||||
|
|
||||||
**Problem:** Brak tokenów CSRF na formularzach panelu admina. State-changing POST endpointy (create/update/delete) są potencjalnie podatne na ataki CSRF.
|
**Problem:** Brak tokenĂłw CSRF na formularzach panelu admina. State-changing POST endpointy (create/update/delete) sÄ… potencjalnie podatne na ataki CSRF.
|
||||||
|
|
||||||
**Scope:** Dodanie CSRF tokenów do formularzy i walidacji w panelu administracyjnym.
|
**Scope:** Dodanie CSRF tokenĂłw do formularzy i walidacji w panelu administracyjnym.
|
||||||
|
|
||||||
**Reference:** `.paul/codebase/concerns.md` — MEDIUM — Missing CSRF tokens
|
**Reference:** `.paul/codebase/concerns.md` — MEDIUM — Missing CSRF tokens
|
||||||
|
|
||||||
### Phase 6 — IntegrationsRepository split
|
### Phase 6 — IntegrationsRepository split
|
||||||
|
|
||||||
**Problem:** `IntegrationsRepository` ma 875 linii — miesza logikę generyczną (settings, logi, product linking) z logiką specyficzną dla Apilo (~650 linii). Narusza zasadę jednej odpowiedzialności.
|
**Problem:** `IntegrationsRepository` ma 875 linii — miesza logikę generyczną (settings, logi, product linking) z logiką specyficzną dla Apilo (~650 linii). Narusza zasadę jednej odpowiedzialności.
|
||||||
|
|
||||||
**Scope:**
|
**Scope:**
|
||||||
- Plan 06-01: Utwórz `ApiloRepository` z metodami apilo* (non-breaking)
|
- Plan 06-01: UtwĂłrz `ApiloRepository` z metodami apilo* (non-breaking)
|
||||||
- Plan 06-02: Zmigruj konsumentów (IntegrationsController, ShopProductController, OrderAdminService, cron.php), usuń apilo* z IntegrationsRepository
|
- Plan 06-02: Zmigruj konsumentów (IntegrationsController, ShopProductController, OrderAdminService, cron.php), usuń apilo* z IntegrationsRepository
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
### Phase 5 — Order bugs fix
|
### Phase 5 — Order bugs fix
|
||||||
|
|
||||||
**Problem 1:** Zduplikowane zamówienia — klient widzi błąd i klika złóż zamówienie ponownie. Pierwsze zamówienie trafiło do bazy mimo błędu. Powrót do `/podsumowanie` regeneruje token i pozwala złożyć drugie zamówienie.
|
**Problem 1:** Zduplikowane zamówienia — klient widzi błąd i klika złóż zamówienie ponownie. Pierwsze zamówienie trafiło do bazy mimo błędu. Powrót do `/podsumowanie` regeneruje token i pozwala złożyć drugie zamówienie.
|
||||||
|
|
||||||
**Problem 2:** Zamówienia COD (płatność przy odbiorze) dostają status "Zamówienie złożone" zamiast "Przyjęte do realizacji". Kod sprawdza hardkodowane `payment_id == 3`, które jest inne w tej instancji sklepu.
|
**Problem 2:** Zamówienia COD (płatność przy odbiorze) dostają status "Zamówienie złożone" zamiast "Przyjęte do realizacji". Kod sprawdza hardkodowane `payment_id == 3`, które jest inne w tej instancji sklepu.
|
||||||
|
|
||||||
**Scope:** Guard w `summaryView()`, try-catch w `basketSave()`, kolumna `is_cod` w `pp_shop_payment_methods`, użycie flagi zamiast hardkodowanego ID.
|
**Scope:** Guard w `summaryView()`, try-catch w `basketSave()`, kolumna `is_cod` w `pp_shop_payment_methods`, uĹĽycie flagi zamiast hardkodowanego ID.
|
||||||
|
|
||||||
---
|
---
|
||||||
*Roadmap created: 2026-03-12*
|
*Roadmap created: 2026-03-12*
|
||||||
### Phase 11 — DataLayer GA4 analytics fix
|
### Phase 11 — DataLayer GA4 analytics fix
|
||||||
|
|
||||||
**Problem:** Eventy dataLayer ecommerce (purchase, begin_checkout, view_item, add_to_cart) używają starego formatu UA (id/name zamiast item_id/item_name), brak currency w view_item, price:0 w purchase, brak eventu view_cart. Remarketing dynamiczny i konwersje GA4 nie działają poprawnie.
|
**Problem:** Eventy dataLayer ecommerce (purchase, begin_checkout, view_item, add_to_cart) używają starego formatu UA (id/name zamiast item_id/item_name), brak currency w view_item, price:0 w purchase, brak eventu view_cart. Remarketing dynamiczny i konwersje GA4 nie działają poprawnie.
|
||||||
|
|
||||||
**Scope:** Poprawka 4 istniejących eventów do formatu GA4 + dodanie nowego eventu view_cart na stronie koszyka.
|
**Scope:** Poprawka 4 istniejÄ…cych eventĂłw do formatu GA4 + dodanie nowego eventu view_cart na stronie koszyka.
|
||||||
|
|
||||||
**Reference:** `poprawki_datalayer_projectpro.md` — audyt analityki z pomysloweprezenty.pl
|
**Reference:** `poprawki_datalayer_projectpro.md` — audyt analityki z pomysloweprezenty.pl
|
||||||
|
|
||||||
### Phase 12 — summaryView redirect fix
|
### Phase 12 — summaryView redirect fix
|
||||||
|
|
||||||
**Problem:** Po złożeniu pierwszego zamówienia, guard w `summaryView()` sprawdzał sesyjny `order-submit-last-order-id` i redirectował na stronę starego zamówienia. Blokował dostęp do `/koszyk-podsumowanie` dla kolejnych zamówień. Poprawka z instancji klienta (change.md) do wdrożenia globalnie.
|
**Problem:** Po złożeniu pierwszego zamówienia, guard w `summaryView()` sprawdzał sesyjny `order-submit-last-order-id` i redirectował na stronę starego zamówienia. Blokował dostęp do `/koszyk-podsumowanie` dla kolejnych zamówień. Poprawka z instancji klienta (change.md) do wdrożenia globalnie.
|
||||||
|
|
||||||
**Scope:** Usunięcie bloku redirect z `summaryView()` w `ShopBasketController.php`. Double-submit protection w `basketSave()` pozostaje bez zmian.
|
**Scope:** Usunięcie bloku redirect z `summaryView()` w `ShopBasketController.php`. Double-submit protection w `basketSave()` pozostaje bez zmian.
|
||||||
|
|
||||||
### Phase 13 — Basket logging + TTL token fix
|
### Phase 13 — Basket logging + TTL token fix
|
||||||
|
|
||||||
**Problem:** Brak logowania w basketSave() uniemożliwia diagnozę błędów zamówień. Token zamówienia jednorazowy — nadpisywany przy każdym wejściu na podsumowanie, co powoduje że druga karta, "wstecz" lub odświeżenie unieważnia formularz.
|
**Problem:** Brak logowania w basketSave() uniemożliwia diagnozę błędów zamówień. Token zamówienia jednorazowy — nadpisywany przy każdym wejściu na podsumowanie, co powoduje że druga karta, "wstecz" lub odświeżenie unieważnia formularz.
|
||||||
|
|
||||||
**Scope:** Dodanie metody logOrder() z 4 punktami logowania, zmiana tokena z jednorazowego na TTL 30 min, redirect przy błędzie tokena na /koszyk-podsumowanie zamiast /koszyk, nowy double-submit guard.
|
**Scope:** Dodanie metody logOrder() z 4 punktami logowania, zmiana tokena z jednorazowego na TTL 30 min, redirect przy błędzie tokena na /koszyk-podsumowanie zamiast /koszyk, nowy double-submit guard.
|
||||||
|
|
||||||
### Phase 14 — Custom fields delete bug
|
### Phase 14 — Custom fields delete bug
|
||||||
|
|
||||||
**Problem:** Usunięcie WSZYSTKICH dodatkowych pól z produktu nie działa. jQuery `.serialize()` nie wysyła klucza `custom_field_name[]` gdy nie ma żadnych pól → `array_key_exists('custom_field_name', $d)` w ProductRepository zwraca false → `saveCustomFields()` nigdy nie jest wywoływany → pola pozostają w bazie.
|
**Problem:** Usunięcie WSZYSTKICH dodatkowych pól z produktu nie działa. jQuery `.serialize()` nie wysyła klucza `custom_field_name[]` gdy nie ma żadnych pól → `array_key_exists('custom_field_name', $d)` w ProductRepository zwraca false → `saveCustomFields()` nigdy nie jest wywoływany → pola pozostają w bazie.
|
||||||
|
|
||||||
**Scope:** Dodanie hidden markera `custom_field_name_present` w szablonie JS + zmiana warunku w ProductRepository na sprawdzanie tego markera. Test jednostkowy.
|
**Scope:** Dodanie hidden markera `custom_field_name_present` w szablonie JS + zmiana warunku w ProductRepository na sprawdzanie tego markera. Test jednostkowy.
|
||||||
|
|
||||||
|
### Phase 15 - Scontainers edit saves as new record
|
||||||
|
|
||||||
|
**Problem:** Edycja kontenera statycznego (`/admin/scontainers/edit/id={id}`) zapisuje rekord jako nowy wpis zamiast aktualizacji. W praktyce podczas zapisu gubi sie `id` i repository wykonuje insert.
|
||||||
|
|
||||||
|
**Scope:** Poprawic przekazywanie `id` w nowym flow formularza ScontainersController + dodac test regresyjny dla edycji, bez zmian globalnych w innych kontrolerach.
|
||||||
|
|
||||||
---
|
---
|
||||||
*Last updated: 2026-04-16*
|
*Last updated: 2026-04-18*
|
||||||
|
|
||||||
|
|||||||
@@ -1,66 +1,72 @@
|
|||||||
# Project State
|
# Project State
|
||||||
|
|
||||||
## Project Reference
|
## Project Reference
|
||||||
|
|
||||||
See: .paul/PROJECT.md (updated 2026-03-12)
|
See: .paul/PROJECT.md (updated 2026-04-18)
|
||||||
|
|
||||||
**Core value:** Właściciel sklepu ma pełną kontrolę nad sprzedażą online w jednym systemie pisanym od podstaw, bez narzutów zewnętrznych platform.
|
**Core value:** Właściciel sklepu ma pełną kontrolę nad sprzedażą online w jednym systemie pisanym od podstaw, bez narzutów zewnętrznych platform.
|
||||||
**Current focus:** Phase 14 complete — custom fields delete bug fix
|
**Current focus:** Phase 15 complete - loop closed (scontainers edit save fix)
|
||||||
|
|
||||||
## Current Position
|
## Current Position
|
||||||
|
|
||||||
Milestone: Hotfix
|
Milestone: Hotfix
|
||||||
Phase: 14 — custom fields delete bug — Complete
|
Phase: 15 of 15 (Scontainers edit save fix) - Complete
|
||||||
Plan: 14-01 complete
|
Plan: 15-01 complete
|
||||||
Status: UNIFY complete, phase 14 finished
|
Status: UNIFY complete, ready for next planning loop
|
||||||
Last activity: 2026-04-16 — 14-01 UNIFY complete
|
Last activity: 2026-04-18 - Closed loop for .paul/phases/15-scontainers-edit-save-fix/15-01-PLAN.md
|
||||||
|
|
||||||
Progress:
|
Progress:
|
||||||
- Phase 14: [██████████] 100% (COMPLETE)
|
- Milestone: [##########] 100%
|
||||||
|
- Phase 15: [##########] 100%
|
||||||
|
|
||||||
## Loop Position
|
## Loop Position
|
||||||
|
|
||||||
Current loop state (phase 14, plan 01):
|
Current loop state:
|
||||||
```
|
```
|
||||||
PLAN ──▶ APPLY ──▶ UNIFY
|
PLAN --> APPLY --> UNIFY
|
||||||
✓ ✓ ✓ [Phase 14 complete]
|
✓ ✓ ✓ [Loop complete - ready for next PLAN]
|
||||||
```
|
```
|
||||||
|
|
||||||
Previous phases:
|
Previous phases:
|
||||||
```
|
```
|
||||||
Phase 4: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-12]
|
Phase 4: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-12]
|
||||||
Phase 5: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-12]
|
Phase 5: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-12]
|
||||||
Phase 6: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-12]
|
Phase 6: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-12]
|
||||||
Phase 7: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-15]
|
Phase 7: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-15]
|
||||||
Phase 8: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-16]
|
Phase 8: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-16]
|
||||||
Phase 9: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-19]
|
Phase 9: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-19]
|
||||||
Phase 10: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-19]
|
Phase 10: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-19]
|
||||||
Phase 11: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-25]
|
Phase 11: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-25]
|
||||||
Phase 12: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-25]
|
Phase 12: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-25]
|
||||||
Phase 13: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-03-25]
|
Phase 13: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-03-25]
|
||||||
Phase 14: PLAN ──▶ APPLY ──▶ UNIFY ✓ ✓ ✓ [COMPLETE — 2026-04-16]
|
Phase 14: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-04-16]
|
||||||
|
Phase 15: PLAN --> APPLY --> UNIFY ✓ ✓ ✓ [COMPLETE - 2026-04-18]
|
||||||
```
|
```
|
||||||
|
|
||||||
## Accumulated Context
|
## Accumulated Context
|
||||||
|
|
||||||
### Decisions
|
### Decisions
|
||||||
|
- 2026-04-18: Transition-phase git commit step pending (not executed during this UNIFY run)
|
||||||
|
- 2026-04-18: Phase 15 loop closed with SUMMARY at .paul/phases/15-scontainers-edit-save-fix/15-01-SUMMARY.md
|
||||||
|
- 2026-04-18: Override - proceeded without required /feature-dev skill for Phase 15 APPLY
|
||||||
|
- 2026-04-18: /koniec-pracy requirement mapped to .claude/commands/koniec-pracy.md guidance for end-of-session release flow
|
||||||
|
- 2026-04-18: Scontainers edit fix - ID from tabbed form can be omitted when hidden field is defined as FormField and not as hiddenFields
|
||||||
- Use existing `CouponRepository::markAsUsed()` instead of adding methods to stdClass
|
- Use existing `CouponRepository::markAsUsed()` instead of adding methods to stdClass
|
||||||
- 2026-03-16: Przyczyna braku wysyłki = brakujące $apiloRepository w use() closures cron.php (regresja z fazy 6)
|
- 2026-03-16: Przyczyna braku wysyłki = brakujące $apiloRepository w use() closures cron.php (regresja z fazy 6)
|
||||||
- 2026-03-16: Retry -1 orders co 1h zamiast permanent failure
|
- 2026-03-16: Retry -1 orders co 1h zamiast permanent failure
|
||||||
- 2026-03-16: Email notification o trwale failed Apilo jobach
|
- 2026-03-16: Email notification o trwale failed Apilo jobach
|
||||||
- 2026-03-19: Order-related Apilo joby — infinite retry co 30 min (nigdy permanent failure)
|
- 2026-03-19: Order-related Apilo joby — infinite retry co 30 min (nigdy permanent failure)
|
||||||
- 2026-03-19: Email z danymi zamówienia + rozróżnienie PONAWIANY vs TRWAŁY BŁĄD
|
- 2026-03-19: Email z danymi zamĂłwienia + rozróżnienie PONAWIANY vs TRWAĹY BĹÄ„D
|
||||||
- 2026-03-19: Cleanup stuck sync_payment/sync_status jobów po udanym wysłaniu
|
- 2026-03-19: Cleanup stuck sync_payment/sync_status jobów po udanym wysłaniu
|
||||||
- 2026-03-19: Edycja custom fields w koszyku — product_code przeliczany po zmianie, merge duplikatów przy identycznym hashu
|
- 2026-03-19: Edycja custom fields w koszyku — product_code przeliczany po zmianie, merge duplikatów przy identycznym hashu
|
||||||
- 2026-03-19: JS handlery koszyka w basket.php (nie basket-details.php) bo basket-details jest AJAX-replaceable
|
- 2026-03-19: JS handlery koszyka w basket.php (nie basket-details.php) bo basket-details jest AJAX-replaceable
|
||||||
- 2026-03-25: view_cart event w basket.php (nie basket-details.php) — ten sam powód
|
- 2026-03-25: view_cart event w basket.php (nie basket-details.php) — ten sam powód
|
||||||
- 2026-03-25: GA4 item format standard: item_id (string), item_name, price (number), quantity (int), google_business_vertical: "retail"
|
- 2026-03-25: GA4 item format standard: item_id (string), item_name, price (number), quantity (int), google_business_vertical: "retail"
|
||||||
- 2026-03-25: Brak user_data w purchase — wymaga analizy RODO
|
- 2026-03-25: Brak user_data w purchase — wymaga analizy RODO
|
||||||
- 2026-03-25: summaryView() redirect guard usunięty — blokował kolejne zamówienia po pierwszym (z change.md instancji klienta)
|
- 2026-03-25: summaryView() redirect guard usunięty — blokował kolejne zamówienia po pierwszym (z change.md instancji klienta)
|
||||||
- 2026-03-25: Token zamówienia z jednorazowego na TTL 30 min — backward compat z plain string
|
- 2026-03-25: Token zamówienia z jednorazowego na TTL 30 min — backward compat z plain string
|
||||||
- 2026-03-25: logOrder() — logowanie błędów zamówień do logs/logs-order-YYYY-MM-DD.log
|
- 2026-03-25: logOrder() — logowanie błędów zamówień do logs/logs-order-YYYY-MM-DD.log
|
||||||
- 2026-03-25: Redirect przy złym tokenie: /koszyk-podsumowanie zamiast /koszyk
|
- 2026-03-25: Redirect przy złym tokenie: /koszyk-podsumowanie zamiast /koszyk
|
||||||
- 2026-04-16: Custom fields delete fix — hidden marker `custom_field_name_present` zamiast `array_key_exists('custom_field_name')`
|
- 2026-04-16: Custom fields delete fix — hidden marker `custom_field_name_present` zamiast `array_key_exists('custom_field_name')`
|
||||||
|
|
||||||
### Deferred Issues
|
### Deferred Issues
|
||||||
None.
|
None.
|
||||||
@@ -68,12 +74,18 @@ None.
|
|||||||
### Blockers/Concerns
|
### Blockers/Concerns
|
||||||
None.
|
None.
|
||||||
|
|
||||||
|
### Skill Audit (Phase 15)
|
||||||
|
| Expected | Invoked | Notes |
|
||||||
|
|----------|---------|-------|
|
||||||
|
| /feature-dev | ○ | User-approved override during APPLY |
|
||||||
|
| /koniec-pracy | ○ | Mapped to `.claude/commands/koniec-pracy.md`; release flow not executed in this loop |
|
||||||
|
|
||||||
## Session Continuity
|
## Session Continuity
|
||||||
|
|
||||||
Last session: 2026-04-16
|
Last session: 2026-04-18
|
||||||
Stopped at: Phase 14 UNIFY complete
|
Stopped at: Phase 15 complete, loop closed
|
||||||
Next action: /koniec-pracy or next feature
|
Next action: Start next work with $paul-plan (or run /koniec-pracy for release flow)
|
||||||
Resume file: .paul/phases/14-custom-fields-delete-bug/14-01-SUMMARY.md
|
Resume file: .paul/phases/15-scontainers-edit-save-fix/15-01-SUMMARY.md
|
||||||
|
|
||||||
---
|
---
|
||||||
*STATE.md — Updated after every significant action*
|
*STATE.md — Updated after every significant action*
|
||||||
|
|
||||||
|
|||||||
13
.paul/changelog/2026-04-18.md
Normal file
13
.paul/changelog/2026-04-18.md
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# 2026-04-18
|
||||||
|
|
||||||
|
## Co zrobiono
|
||||||
|
|
||||||
|
- [Phase 15, Plan 01] Naprawiono regresje zapisu edycji kontenerow statycznych (update zamiast tworzenia nowego rekordu).
|
||||||
|
- Przeniesiono przekazywanie `id` w formularzu Scontainers do `hiddenFields` oraz dodano fallback `id` z route parametru.
|
||||||
|
- Dodano testy regresyjne dla mapowania `id` i create-flow w `ScontainersControllerTest`.
|
||||||
|
|
||||||
|
## Zmienione pliki
|
||||||
|
|
||||||
|
- `autoload/admin/Controllers/ScontainersController.php`
|
||||||
|
- `tests/Unit/admin/Controllers/ScontainersControllerTest.php`
|
||||||
|
- `.paul/phases/15-scontainers-edit-save-fix/15-01-SUMMARY.md`
|
||||||
157
.paul/phases/15-scontainers-edit-save-fix/15-01-PLAN.md
Normal file
157
.paul/phases/15-scontainers-edit-save-fix/15-01-PLAN.md
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
---
|
||||||
|
phase: 15-scontainers-edit-save-fix
|
||||||
|
plan: 01
|
||||||
|
type: execute
|
||||||
|
wave: 1
|
||||||
|
depends_on: []
|
||||||
|
files_modified:
|
||||||
|
- autoload/admin/Controllers/ScontainersController.php
|
||||||
|
- tests/Unit/admin/Controllers/ScontainersControllerTest.php
|
||||||
|
autonomous: true
|
||||||
|
delegation: off
|
||||||
|
---
|
||||||
|
|
||||||
|
<objective>
|
||||||
|
## Goal
|
||||||
|
Naprawic regresje w edycji kontenerow statycznych: zapis edytowanego rekordu nie moze tworzyc nowego wpisu.
|
||||||
|
|
||||||
|
## Purpose
|
||||||
|
Administrator musi miec pewnosc, ze edycja kontenera aktualizuje istniejace ID. Obecny blad powoduje duplikaty i ryzyko niespojnych tresci.
|
||||||
|
|
||||||
|
## Output
|
||||||
|
- Poprawiony flow zapisu w `ScontainersController`, ktory zawsze przekazuje poprawne `id` przy edycji
|
||||||
|
- Testy jednostkowe zabezpieczajace przed powrotem regresji
|
||||||
|
</objective>
|
||||||
|
|
||||||
|
<context>
|
||||||
|
## Project Context
|
||||||
|
@.paul/PROJECT.md
|
||||||
|
@.paul/ROADMAP.md
|
||||||
|
@.paul/STATE.md
|
||||||
|
|
||||||
|
## Source Files
|
||||||
|
@autoload/admin/Controllers/ScontainersController.php
|
||||||
|
@admin/templates/components/form-edit.php
|
||||||
|
@autoload/admin/ViewModels/Forms/FormEditViewModel.php
|
||||||
|
@autoload/admin/Support/Forms/FormRequestHandler.php
|
||||||
|
@tests/Unit/admin/Controllers/ScontainersControllerTest.php
|
||||||
|
</context>
|
||||||
|
|
||||||
|
<skills>
|
||||||
|
## Required Skills (from SPECIAL-FLOWS.md)
|
||||||
|
|
||||||
|
| Skill | Priority | When to Invoke | Loaded? |
|
||||||
|
|-------|----------|----------------|---------|
|
||||||
|
| /feature-dev | required | Before implementation in APPLY | ○ |
|
||||||
|
| /koniec-pracy | required | After implementation/release wrap-up | ○ |
|
||||||
|
|
||||||
|
**BLOCKING:** Required skills MUST be loaded before APPLY proceeds.
|
||||||
|
Run each skill command or confirm already loaded.
|
||||||
|
|
||||||
|
## Skill Invocation Checklist
|
||||||
|
- [ ] /feature-dev loaded (run command or confirm)
|
||||||
|
- [ ] /koniec-pracy loaded (run command or confirm)
|
||||||
|
|
||||||
|
</skills>
|
||||||
|
|
||||||
|
<acceptance_criteria>
|
||||||
|
|
||||||
|
## AC-1: Edycja nie tworzy nowego kontenera
|
||||||
|
```gherkin
|
||||||
|
Given istnieje kontener statyczny o ID 9
|
||||||
|
When admin wejdzie w /admin/scontainers/edit/id=9 i kliknie "Zatwierdz"
|
||||||
|
Then rekord o ID 9 zostanie zaktualizowany
|
||||||
|
And nie powstanie nowy rekord w pp_scontainers
|
||||||
|
```
|
||||||
|
|
||||||
|
## AC-2: Tworzenie nowego kontenera nadal dziala
|
||||||
|
```gherkin
|
||||||
|
Given admin otwiera /admin/scontainers/edit/ bez ID
|
||||||
|
When wypelni dane i kliknie "Zatwierdz"
|
||||||
|
Then zapis utworzy nowy rekord w pp_scontainers
|
||||||
|
```
|
||||||
|
|
||||||
|
## AC-3: API legacy JSON pozostaje bez zmian
|
||||||
|
```gherkin
|
||||||
|
Given zapis kontenera odbywa sie przez legacy payload values (JSON)
|
||||||
|
When wywolywana jest sciezka legacy w ScontainersController::save()
|
||||||
|
Then zachowanie insert/update pozostaje zgodne z dotychczasowa logika
|
||||||
|
```
|
||||||
|
|
||||||
|
</acceptance_criteria>
|
||||||
|
|
||||||
|
<tasks>
|
||||||
|
|
||||||
|
<task type="auto">
|
||||||
|
<name>Task 1: Utrwalic przekazywanie ID w nowym formularzu scontainers</name>
|
||||||
|
<files>autoload/admin/Controllers/ScontainersController.php</files>
|
||||||
|
<action>
|
||||||
|
W `buildFormViewModel()` przeniesc `id` do `hiddenFields` (FormEditViewModel),
|
||||||
|
tak aby pole `id` bylo renderowane niezaleznie od zakladek.
|
||||||
|
|
||||||
|
W `save()` dodac defensywny fallback: jesli `data['id']` z requestu jest puste,
|
||||||
|
pobrac `id` z parametru trasy (`Helpers::get('id')`) i uzyc go przy zapisie.
|
||||||
|
|
||||||
|
Nie zmieniac flow legacy (`values` JSON) ani logiki repozytorium.
|
||||||
|
</action>
|
||||||
|
<verify>Manual check: edycja /admin/scontainers/edit/id=9 aktualizuje rekord 9 zamiast tworzyc nowy</verify>
|
||||||
|
<done>AC-1 i AC-3 satisfied</done>
|
||||||
|
</task>
|
||||||
|
|
||||||
|
<task type="auto">
|
||||||
|
<name>Task 2: Dodac test regresyjny dla formularza i mapowania ID</name>
|
||||||
|
<files>tests/Unit/admin/Controllers/ScontainersControllerTest.php</files>
|
||||||
|
<action>
|
||||||
|
Rozszerzyc testy kontrolera o przypadki potwierdzajace, ze formularz edycji
|
||||||
|
niesie `id` jako hidden field oraz ze flow zapisu potrafi odczytac ID rekordu
|
||||||
|
dla przypadku edycji.
|
||||||
|
|
||||||
|
Uzyc Reflection tam, gdzie potrzeba dostepu do prywatnych metod (zgodnie z obecnym stylem testow).
|
||||||
|
</action>
|
||||||
|
<verify>./test.ps1 tests/Unit/admin/Controllers/ScontainersControllerTest.php</verify>
|
||||||
|
<done>AC-1 covered by automated tests</done>
|
||||||
|
</task>
|
||||||
|
|
||||||
|
<task type="auto">
|
||||||
|
<name>Task 3: Zweryfikowac brak regresji create flow</name>
|
||||||
|
<files>autoload/admin/Controllers/ScontainersController.php, tests/Unit/admin/Controllers/ScontainersControllerTest.php</files>
|
||||||
|
<action>
|
||||||
|
Potwierdzic, ze nowy kontener (brak `id` w URL i formularzu) nadal tworzy nowy rekord.
|
||||||
|
Dostosowac warunki fallbacku tak, by nie wymuszaly update przy create.
|
||||||
|
</action>
|
||||||
|
<verify>Manual check: /admin/scontainers/edit/ -> Zatwierdz tworzy nowe ID</verify>
|
||||||
|
<done>AC-2 satisfied</done>
|
||||||
|
</task>
|
||||||
|
|
||||||
|
</tasks>
|
||||||
|
|
||||||
|
<boundaries>
|
||||||
|
|
||||||
|
## DO NOT CHANGE
|
||||||
|
- autoload/Domain/Scontainers/ScontainersRepository.php (brak zmian logiki insert/update na poziomie repo)
|
||||||
|
- admin/templates/components/form-edit.php (bez globalnych zmian w uniwersalnym komponencie)
|
||||||
|
- Inne kontrolery admin poza ScontainersController
|
||||||
|
|
||||||
|
## SCOPE LIMITS
|
||||||
|
- Zakres tylko dla problemu edycji kontenerow statycznych (scontainers)
|
||||||
|
- Bez refaktoryzacji calego systemu FormEdit
|
||||||
|
|
||||||
|
</boundaries>
|
||||||
|
|
||||||
|
<verification>
|
||||||
|
Before declaring plan complete:
|
||||||
|
- [ ] ./test.ps1 tests/Unit/admin/Controllers/ScontainersControllerTest.php
|
||||||
|
- [ ] Manual: edycja istniejacego kontenera nie tworzy nowego rekordu
|
||||||
|
- [ ] Manual: tworzenie nowego kontenera nadal dziala
|
||||||
|
- [ ] All acceptance criteria met
|
||||||
|
</verification>
|
||||||
|
|
||||||
|
<success_criteria>
|
||||||
|
- Blad edycji kontenerow statycznych nie wystepuje
|
||||||
|
- Test regresyjny przechodzi
|
||||||
|
- Brak regresji w create flow dla scontainers
|
||||||
|
</success_criteria>
|
||||||
|
|
||||||
|
<output>
|
||||||
|
After completion, create `.paul/phases/15-scontainers-edit-save-fix/15-01-SUMMARY.md`
|
||||||
|
</output>
|
||||||
108
.paul/phases/15-scontainers-edit-save-fix/15-01-SUMMARY.md
Normal file
108
.paul/phases/15-scontainers-edit-save-fix/15-01-SUMMARY.md
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
---
|
||||||
|
phase: 15-scontainers-edit-save-fix
|
||||||
|
plan: 01
|
||||||
|
subsystem: admin
|
||||||
|
tags: [scontainers, form-edit, hidden-fields, regression-fix]
|
||||||
|
|
||||||
|
requires: []
|
||||||
|
provides:
|
||||||
|
- Fix edycji scontainers (update zamiast insert)
|
||||||
|
- Regresyjne testy kontrolera dla mapowania id
|
||||||
|
affects: []
|
||||||
|
|
||||||
|
tech-stack:
|
||||||
|
added: []
|
||||||
|
patterns: [hiddenFields for stable id transfer in tabbed form-edit]
|
||||||
|
|
||||||
|
key-files:
|
||||||
|
created: []
|
||||||
|
modified:
|
||||||
|
- autoload/admin/Controllers/ScontainersController.php
|
||||||
|
- tests/Unit/admin/Controllers/ScontainersControllerTest.php
|
||||||
|
|
||||||
|
key-decisions:
|
||||||
|
- "Przeniesienie id z FormField::hidden do hiddenFields w FormEditViewModel"
|
||||||
|
- "Fallback id z route parametru przy zapisie edycji"
|
||||||
|
|
||||||
|
patterns-established:
|
||||||
|
- "W formularzach z zakladkami id encji przekazujemy przez hiddenFields, nie przez pola przypisane do taba"
|
||||||
|
|
||||||
|
duration: ~20min
|
||||||
|
completed: 2026-04-18
|
||||||
|
---
|
||||||
|
|
||||||
|
# Phase 15 Plan 01: Scontainers edit save fix - Summary
|
||||||
|
|
||||||
|
**Naprawiono regresje, przez ktora edycja kontenera statycznego tworzyla nowy rekord zamiast aktualizacji istniejacego ID.**
|
||||||
|
|
||||||
|
## Performance
|
||||||
|
|
||||||
|
| Metric | Value |
|
||||||
|
|--------|-------|
|
||||||
|
| Duration | ~20min |
|
||||||
|
| Completed | 2026-04-18 |
|
||||||
|
| Tasks | 3 completed |
|
||||||
|
| Files modified | 2 |
|
||||||
|
|
||||||
|
## Acceptance Criteria Results
|
||||||
|
|
||||||
|
| Criterion | Status | Notes |
|
||||||
|
|-----------|--------|-------|
|
||||||
|
| AC-1: Edycja nie tworzy nowego kontenera | Pass | `id` jest zawsze przenoszone przez hiddenFields + fallback z URL przy braku w POST |
|
||||||
|
| AC-2: Tworzenie nowego kontenera nadal dziala | Pass | Dla create `id=0`, action pozostaje `/admin/scontainers/save/` |
|
||||||
|
| AC-3: API legacy JSON pozostaje bez zmian | Pass | Sciezka `values` (legacy) nie byla modyfikowana |
|
||||||
|
|
||||||
|
## Accomplishments
|
||||||
|
|
||||||
|
- Przeniesiono `id` do `hiddenFields` w `ScontainersController::buildFormViewModel()`, co eliminuje gubienie `id` w formularzu tabowanym.
|
||||||
|
- Dodano defensywny fallback na `id` z parametru trasy w `ScontainersController::save()`.
|
||||||
|
- Dodano 2 testy regresyjne dla mapowania `id` i create-flow.
|
||||||
|
|
||||||
|
## Files Created/Modified
|
||||||
|
|
||||||
|
| File | Change | Purpose |
|
||||||
|
|------|--------|---------|
|
||||||
|
| `autoload/admin/Controllers/ScontainersController.php` | Modified | Stabilne przekazywanie `id` dla update oraz fallback route `id` |
|
||||||
|
| `tests/Unit/admin/Controllers/ScontainersControllerTest.php` | Modified | Testy regresyjne dla hiddenFields i create flow |
|
||||||
|
|
||||||
|
## Decisions Made
|
||||||
|
|
||||||
|
| Decision | Rationale | Impact |
|
||||||
|
|----------|-----------|--------|
|
||||||
|
| Uzyc `hiddenFields` zamiast `FormField::hidden('id')` | Hidden field w tabbed form moze nie byc renderowany w aktywnej strukturze pol | Brak tworzenia duplikatow przy edycji |
|
||||||
|
| Dodac fallback `id` z URL w `save()` | Dodatkowa odpornosc na brak `id` w payloadzie | Bezpieczny update dla `/admin/scontainers/save/id={id}` |
|
||||||
|
|
||||||
|
## Deviations from Plan
|
||||||
|
|
||||||
|
Brak istotnych odchylen implementacyjnych.
|
||||||
|
|
||||||
|
Skill audit:
|
||||||
|
- `/feature-dev` - pominiety na prosbe uzytkownika (override zapisany w STATE.md)
|
||||||
|
- `/koniec-pracy` - wymaganie zmapowane na `.claude/commands/koniec-pracy.md`
|
||||||
|
|
||||||
|
## Issues Encountered
|
||||||
|
|
||||||
|
| Issue | Resolution |
|
||||||
|
|-------|------------|
|
||||||
|
| Brak `test.ps1` w workspace | Test uruchomiony bezposrednio przez `php phpunit.phar ...` |
|
||||||
|
|
||||||
|
## Verification Results
|
||||||
|
|
||||||
|
- `php phpunit.phar tests/Unit/admin/Controllers/ScontainersControllerTest.php`
|
||||||
|
- Wynik: `OK (6 tests, 20 assertions)`
|
||||||
|
|
||||||
|
## Next Phase Readiness
|
||||||
|
|
||||||
|
**Ready:**
|
||||||
|
- Problem zapisu scontainers naprawiony na poziomie kontrolera.
|
||||||
|
- Testy regresyjne zabezpieczaja krytyczny przypadek.
|
||||||
|
|
||||||
|
**Concerns:**
|
||||||
|
- Manualna weryfikacja UI edycji/create nadal wskazana po stronie panelu admin.
|
||||||
|
|
||||||
|
**Blockers:**
|
||||||
|
- None.
|
||||||
|
|
||||||
|
---
|
||||||
|
*Phase: 15-scontainers-edit-save-fix, Plan: 01*
|
||||||
|
*Completed: 2026-04-18*
|
||||||
@@ -55,7 +55,7 @@ composer test # standard
|
|||||||
|
|
||||||
PHPUnit 9.6 via `phpunit.phar`. Bootstrap: `tests/bootstrap.php`. Config: `phpunit.xml`.
|
PHPUnit 9.6 via `phpunit.phar`. Bootstrap: `tests/bootstrap.php`. Config: `phpunit.xml`.
|
||||||
|
|
||||||
Current suite: **821 tests, 2278 assertions**.
|
Current suite: **823 tests, 2284 assertions**.
|
||||||
|
|
||||||
### Creating Updates
|
### Creating Updates
|
||||||
See `docs/UPDATE_INSTRUCTIONS.md` for the full procedure. Updates are ZIP packages in `updates/0.XX/`. Never include `*.md` files, `updates/changelog.php`, or root `.htaccess` in update ZIPs. ZIP structure must start directly from project directories — no version subfolder inside the archive.
|
See `docs/UPDATE_INSTRUCTIONS.md` for the full procedure. Updates are ZIP packages in `updates/0.XX/`. Never include `*.md` files, `updates/changelog.php`, or root `.htaccess` in update ZIPs. ZIP structure must start directly from project directories — no version subfolder inside the archive.
|
||||||
@@ -243,4 +243,4 @@ Before starting implementation, review current state of docs.
|
|||||||
|
|
||||||
## Za każdym razem jak próbujesz sprawdzić jakiś plik z logami spróbuj go najpierw pobrać z serwera FTP
|
## Za każdym razem jak próbujesz sprawdzić jakiś plik z logami spróbuj go najpierw pobrać z serwera FTP
|
||||||
|
|
||||||
## Wszystkie pliki które tworzysz jako pomocnicze, np build_0330.ps1 czy build-update.ps1 twórz w folderze temp
|
## Wszystkie pliki które tworzysz jako pomocnicze, np build_0330.ps1 czy build-update.ps1 twórz w folderze temp
|
||||||
|
|||||||
@@ -184,8 +184,16 @@ class ScontainersController
|
|||||||
}
|
}
|
||||||
|
|
||||||
$data = $result['data'];
|
$data = $result['data'];
|
||||||
|
$containerId = (int)($data['id'] ?? 0);
|
||||||
|
if ($containerId <= 0) {
|
||||||
|
$routeId = (int)\Shared\Helpers\Helpers::get('id');
|
||||||
|
if ($routeId > 0) {
|
||||||
|
$containerId = $routeId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$savedId = $this->repository->save([
|
$savedId = $this->repository->save([
|
||||||
'id' => (int)($data['id'] ?? 0),
|
'id' => $containerId,
|
||||||
'status' => $data['status'] ?? 0,
|
'status' => $data['status'] ?? 0,
|
||||||
'show_title' => $data['show_title'] ?? 0,
|
'show_title' => $data['show_title'] ?? 0,
|
||||||
'translations' => $data['translations'] ?? [],
|
'translations' => $data['translations'] ?? [],
|
||||||
@@ -240,7 +248,6 @@ class ScontainersController
|
|||||||
];
|
];
|
||||||
|
|
||||||
$fields = [
|
$fields = [
|
||||||
FormField::hidden('id', $id),
|
|
||||||
FormField::langSection('translations', 'content', [
|
FormField::langSection('translations', 'content', [
|
||||||
FormField::text('title', [
|
FormField::text('title', [
|
||||||
'label' => 'Tytul',
|
'label' => 'Tytul',
|
||||||
@@ -283,7 +290,7 @@ class ScontainersController
|
|||||||
$actionUrl,
|
$actionUrl,
|
||||||
'/admin/scontainers/list/',
|
'/admin/scontainers/list/',
|
||||||
true,
|
true,
|
||||||
[],
|
['id' => $id],
|
||||||
$languages,
|
$languages,
|
||||||
$errors
|
$errors
|
||||||
);
|
);
|
||||||
|
|||||||
1055
docs/CHANGELOG.md
1055
docs/CHANGELOG.md
File diff suppressed because it is too large
Load Diff
@@ -23,10 +23,10 @@ composer test # standard
|
|||||||
## Aktualny stan
|
## Aktualny stan
|
||||||
|
|
||||||
```text
|
```text
|
||||||
OK (821 tests, 2278 assertions)
|
OK (823 tests, 2284 assertions)
|
||||||
```
|
```
|
||||||
|
|
||||||
Zweryfikowano: 2026-04-16 (ver. 0.346)
|
Zweryfikowano: 2026-04-18 (ver. 0.347)
|
||||||
|
|
||||||
## Konfiguracja
|
## Konfiguracja
|
||||||
|
|
||||||
|
|||||||
3496
docs/TODO.md
3496
docs/TODO.md
File diff suppressed because it is too large
Load Diff
@@ -55,5 +55,38 @@ class ScontainersControllerTest extends TestCase
|
|||||||
$this->assertEquals('Domain\Scontainers\ScontainersRepository', $params[0]->getType()->getName());
|
$this->assertEquals('Domain\Scontainers\ScontainersRepository', $params[0]->getType()->getName());
|
||||||
$this->assertEquals('Domain\Languages\LanguagesRepository', $params[1]->getType()->getName());
|
$this->assertEquals('Domain\Languages\LanguagesRepository', $params[1]->getType()->getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testBuildFormViewModelStoresIdInHiddenFieldsForEdit(): void
|
||||||
|
{
|
||||||
|
$reflection = new \ReflectionClass(ScontainersController::class);
|
||||||
|
$method = $reflection->getMethod('buildFormViewModel');
|
||||||
|
$method->setAccessible(true);
|
||||||
|
|
||||||
|
$container = [
|
||||||
|
'id' => 9,
|
||||||
|
'status' => 1,
|
||||||
|
'show_title' => 1,
|
||||||
|
'languages' => [],
|
||||||
|
];
|
||||||
|
|
||||||
|
$form = $method->invoke($this->controller, $container, [], null);
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('id', $form->hiddenFields);
|
||||||
|
$this->assertSame(9, (int)$form->hiddenFields['id']);
|
||||||
|
$this->assertSame('/admin/scontainers/save/id=9', $form->action);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testBuildFormViewModelKeepsCreateFlowWithZeroId(): void
|
||||||
|
{
|
||||||
|
$reflection = new \ReflectionClass(ScontainersController::class);
|
||||||
|
$method = $reflection->getMethod('buildFormViewModel');
|
||||||
|
$method->setAccessible(true);
|
||||||
|
|
||||||
|
$form = $method->invoke($this->controller, [], [], null);
|
||||||
|
|
||||||
|
$this->assertArrayHasKey('id', $form->hiddenFields);
|
||||||
|
$this->assertSame(0, (int)$form->hiddenFields['id']);
|
||||||
|
$this->assertSame('/admin/scontainers/save/', $form->action);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user