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:
parent
04b591fa9e
commit
7444f90d6a
16 changed files with 360 additions and 202 deletions
|
|
@ -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:
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue