--- phase: 02-jawnosc-cen plan: 01 type: execute wave: 2 depends_on: ["01-01"] files_modified: - wp-content/plugins/elementor-addon/elementor-addon.php autonomous: false --- ## Goal Zbudować publiczny XML endpoint z danymi cenowymi apartamentów oraz XML katalog zgodny z XSD portalu dane.gov.pl — gotowy do automatycznego zasilania Ministerstwa. ## Purpose Ustawa o jawności cen zobowiązuje deweloperów do raportowania cen mieszkań do portalu rządowego dane.gov.pl. Deweloper musi podać Ministerstwu URL pliku XML, który portal będzie cyklicznie pobierał. Budujemy te endpointy po stronie WordPress, korzystając z istniejącej tabeli `wp_price_history` i pól ACF. ## Output - `/ceny-mieszkan.xml` — plik z cenami wszystkich lokali + historia z DB - `/ceny-mieszkan.md5` — hash MD5 dla powyższego - `/dane-gov-pl.xml` — katalog XSD-compliant dla dane.gov.pl wskazujący na plik cen - `/dane-gov-pl.md5` — hash MD5 dla katalogu - Strona administracyjna z URL-ami do zgłoszenia do Ministerstwa ## Project Context @.paul/PROJECT.md @.paul/ROADMAP.md @.paul/STATE.md ## Prior Work @.paul/phases/01-historia-cen/01-01-SUMMARY.md ## Source Files @wp-content/plugins/elementor-addon/elementor-addon.php ## AC-1: Endpoint cen mieszkań dostępny ```gherkin Given WordPress jest uruchomiony z przepłukanymi regułami permalink When GET /ceny-mieszkan.xml Then odpowiedź HTTP 200 z Content-Type: application/xml, XML zawiera element z listą dla każdego posta apartamenty, każdy ma: id, nazwa, typ, pietro, powierzchnia, cena_brutto, cena_za_m2, historia_cen ``` ## AC-2: MD5 companions poprawne ```gherkin Given pliki XML są serwowane przez WordPress When GET /ceny-mieszkan.md5 lub /dane-gov-pl.md5 Then odpowiedź HTTP 200 z 32-znakowym lowercase hex stringiem (MD5 odpowiadającego XML) ``` ## AC-3: Katalog dane.gov.pl zgodny z XSD ```gherkin Given endpoint /dane-gov-pl.xml działa When GET /dane-gov-pl.xml Then XML zawiera element z opisującym inwestycję, wskazuje na home_url('/ceny-mieszkan.xml'), updateFrequency to "daily", kategoria to "REGI" ``` ## AC-4: Strona administracyjna z URL-ami ```gherkin Given admin jest zalogowany When odwiedzi Narzędzia → Jawność Cen (wp-admin/tools.php?page=jawnosc-cen) Then widzi oba URL-e do zgłoszenia (ceny-mieszkan.xml i dane-gov-pl.xml), przyciski "Kopiuj URL" i "Otwórz XML", informację o harmonogramie (codziennie) ``` Task 1: Rewrite rules + endpoint cen mieszkań (XML + MD5) wp-content/plugins/elementor-addon/elementor-addon.php Dodaj na końcu pliku (po istniejących hookach) następujące funkcje: **1. Rewrite rules (hook: init, priority 10):** ``` add_rewrite_rule('^ceny-mieszkan\.(xml|md5)$', 'index.php?apartamenty_xml=$matches[1]', 'top'); add_rewrite_rule('^dane-gov-pl\.(xml|md5)$', 'index.php?apartamenty_datagov=$matches[1]', 'top'); ``` Dodaj query vars: 'apartamenty_xml', 'apartamenty_datagov' przez filter 'query_vars'. **2. Generator XML cen (funkcja apartamenty_generate_price_xml()):** - Sprawdź transient 'apartamenty_price_xml_cache' (TTL 1h) — jeśli istnieje, zwróć go - WP_Query: post_type=apartamenty, posts_per_page=-1, orderby=title, order=ASC - Dla każdego postu: - Pobierz: post ID, post_title - get_post_meta flat: information_type, information_floor, information_floor_space, information_price, information_price_m2, information_status - Pobierz historię z wp_price_history: SELECT recorded_at, price, price_m2 WHERE post_id = %d ORDER BY recorded_at DESC - Buduj XML jako string (nie DOMDocument — nie ma pewności że rozszerzenie jest dostępne, użyj SimpleXMLElement lub czystego PHP string z esc_xml/htmlspecialchars): ```xml [post_title] [information_type] [information_floor] [information_floor_space] [information_status] [information_price] [information_price_m2] [max recorded_at from history or today] [price] [price_m2] ... ``` - Zapisz do transientu i zwróć - Unika: nie używaj DOMDocument jeśli brak ext-dom, używaj SimpleXMLElement lub czystych stringów z htmlspecialchars(value, ENT_XML1, 'UTF-8') **3. Endpoint handler (hook: template_redirect, priority 1):** - get_query_var('apartamenty_xml') == 'xml' → header + echo apartamenty_generate_price_xml() + exit - get_query_var('apartamenty_xml') == 'md5' → header Content-Type text/plain + echo md5(apartamenty_generate_price_xml()) + exit **4. Inwalidacja cache w istniejącym cronie apartamenty_record_prices:** Dodaj na początku funkcji crona: delete_transient('apartamenty_price_xml_cache'); Avoid: nie dodawaj flush_rewrite_rules() w runtime (tylko przy aktywacji); nie generuj XML przez DOMDocument bez sprawdzenia extension_loaded('dom'). 1. Przejdź do Ustawienia → Bezpośrednie odnośniki → Zapisz (flush rewrite rules) 2. Otwórz w przeglądarce: [site_url]/ceny-mieszkan.xml 3. Sprawdź: Content-Type w nagłówkach = application/xml, XML jest poprawny (widoczna struktura w przeglądarce) 4. Otwórz: [site_url]/ceny-mieszkan.md5 5. Sprawdź: 32-znakowy lowercase hex string AC-1 i AC-2 (część ceny) spełnione: endpoint XML cen dostępny, MD5 poprawny Task 2: Endpoint katalogu dane.gov.pl (XML + MD5) wp-content/plugins/elementor-addon/elementor-addon.php Dodaj funkcję apartamenty_generate_datagov_xml(): Generuje XML zgodny z XSD https://www.dane.gov.pl/static/xml/otwarte_dane_latest.xsd: ```xml wyszynskiego12-ceny-mieszkan-v1 <polish>Ceny mieszkań – Wyszyńskiego 12</polish> Historia cen lokali mieszkalnych w inwestycji przy ul. Wyszyńskiego 12. Dane aktualizowane codziennie zgodnie z ustawą o jawności cen. daily REGI wyszynskiego12-ceny-xml-v1 [home_url('/ceny-mieszkan.xml')] <polish>Cennik lokali XML</polish> Plik XML z aktualnym cennikiem i historią zmian cen wszystkich lokali remote [date('Y-m-d\T00:00:00.000\Z')] mieszkania ceny deweloper jawność cen historia cen ``` Użyj home_url() dla URL-a zasobu (nie hardcode). Uzupełnij template_redirect handler: - get_query_var('apartamenty_datagov') == 'xml' → header + echo apartamenty_generate_datagov_xml() + exit - get_query_var('apartamenty_datagov') == 'md5' → header text/plain + echo md5(apartamenty_generate_datagov_xml()) + exit Avoid: nie używaj hardcoded URL-a strony; nie pomijaj pola `categories` (wymagane przez XSD). 1. Otwórz: [site_url]/dane-gov-pl.xml 2. Sprawdź: Content-Type application/xml, struktura XML poprawna, URL w wskazuje na /ceny-mieszkan.xml 3. Otwórz: [site_url]/dane-gov-pl.md5 4. Sprawdź: 32-znakowy lowercase hex string AC-2 (komplet) i AC-3 spełnione: oba endpointy XML + oba MD5 działają Task 3: Strona administracyjna Jawność Cen wp-content/plugins/elementor-addon/elementor-addon.php Zarejestruj stronę w menu Narzędzia WP (hook: admin_menu): ```php add_management_page( 'Jawność Cen', 'Jawność Cen', 'manage_options', 'jawnosc-cen', 'apartamenty_jawnosc_cen_page' ); ``` Funkcja apartamenty_jawnosc_cen_page(): - Wyświetl dwa bloki (prostą tabelę HTML, bez CSS frameworków): - Blok 1: "Plik cen (dane)": URL = home_url('/ceny-mieszkan.xml'), [przycisk kopiuj JS], [link "Otwórz"] - Blok 2: "Katalog dane.gov.pl (zgłoś Ministerstwu)": URL = home_url('/dane-gov-pl.xml'), [przycisk kopiuj JS], [link "Otwórz"] - Informacja tekstowa: "Dane aktualizowane codziennie przez WP Cron. Zgłoś URL katalogu dane.gov.pl do administratora portalu: kontakt@dane.gov.pl" - Prosty `