per-step screenshot + html snapshots, matches-only list, full German UI, CSV export

* apply: Recorder.step_snap(page, name) captures both a JPEG screenshot and
  the page HTML for every major moment; every provider now calls step_snap at
  each logical step so failure reports contain the exact DOM and rendered
  state at every stage of the flow
* ZIP report: each snapshot becomes snapshots/NN_<label>.jpg +
  snapshots/NN_<label>.html for AI-assisted debugging
* web: Wohnungsliste zeigt nur noch Flats, die die eigenen Filter treffen;
  Match-Chip entfernt (Liste ist jetzt implizit matchend)
* UI komplett auf Deutsch: Protokoll statt Logs, Administrator statt admin,
  Trockenmodus statt dry-run, Automatik pausiert statt circuit open,
  Alarm statt Alert, Abmelden statt Logout
* Wohnungen-Header: Zeile 1 Info (Alarm + Filter), Zeile 2 Schalter mit
  echten Radio-Paaren (An/Aus) für Automatisch bewerben und Trockenmodus;
  hx-confirm auf den kritischen Radios; per-form CSS für sichtbaren Check-State
* Protokoll: von/bis-Datumsfilter (Berliner Zeit) + CSV-Download
  (/logs/export.csv) mit UTC + lokaler Zeit

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Moritz 2026-04-21 11:40:12 +02:00
parent 04b591fa9e
commit 7444f90d6a
16 changed files with 360 additions and 202 deletions

View file

@ -31,6 +31,7 @@ MAX_CONSOLE_ENTRIES = 200
MAX_NETWORK_ENTRIES = 150
MAX_BODY_SNIPPET = 2000
MAX_HTML_DUMP = 200_000 # 200 KB
MAX_SCREENSHOTS = 40
SCREENSHOT_JPEG_QUALITY = 60
@ -117,21 +118,33 @@ class Recorder:
page.on("request", on_request)
page.on("response", lambda r: asyncio.create_task(on_response(r)))
# --- screenshots --------------------------------------------------------
# --- screenshots + html dump -------------------------------------------
async def snap(self, page, label: str) -> None:
"""Capture screenshot + full page HTML for this moment."""
if len(self.screenshots) >= MAX_SCREENSHOTS:
return
ts = round(time.time() - self.started_at, 3)
entry = {"ts": ts, "label": label, "url": page.url,
"b64_jpeg": "", "size": 0, "html": "", "html_size": 0}
try:
data = await page.screenshot(type="jpeg", quality=SCREENSHOT_JPEG_QUALITY,
full_page=False, timeout=5000)
b64 = base64.b64encode(data).decode("ascii")
self.screenshots.append({
"ts": round(time.time() - self.started_at, 3),
"label": label,
"url": page.url,
"b64_jpeg": b64,
"size": len(data),
})
img = await page.screenshot(type="jpeg", quality=SCREENSHOT_JPEG_QUALITY,
full_page=False, timeout=5000)
entry["b64_jpeg"] = base64.b64encode(img).decode("ascii")
entry["size"] = len(img)
except Exception as e:
logger.warning("snap failed (%s): %s", label, e)
logger.warning("snap screenshot failed (%s): %s", label, e)
try:
html = await page.content()
entry["html"] = html[:MAX_HTML_DUMP]
entry["html_size"] = len(html)
except Exception as e:
logger.warning("snap html failed (%s): %s", label, e)
self.screenshots.append(entry)
async def step_snap(self, page, name: str, detail: str = "", status: str = "ok") -> None:
"""Log a step AND capture a screenshot + HTML for it."""
self.step(name, status, detail)
await self.snap(page, name)
async def finalize(self, page) -> None:
try: