Admin-Wartung: obsolete Buttons entfernt, Enrichment-Status + Foto-Laden hinzugefügt

- Temperament→Deutsch-Button entfernt (0 englische Temperamente in DB)
- Gemma-Einträge-zurücksetzen entfernt (0 Gemma-Einträge, alle Claude Haiku)
- Neuer Button: Enrichment-Status (GET /api/admin/wiki/enrichment-status)
  zeigt Gesamt/Angereichert/Kein-Wiki/Ausstehend/Fotos/Modelle-Verteilung
- Neuer Button: Fotos laden (POST /api/admin/wiki/fetch-photos)
  führt fetch_wiki_images.py --limit 50 aus und zählt gespeicherte Fotos
- SW by-v361, APP_VER 346
This commit is contained in:
rene 2026-04-25 08:53:28 +02:00
parent 0af3078a2a
commit d0921a28e9
4 changed files with 54 additions and 26 deletions

View file

@ -671,23 +671,49 @@ async def wiki_evaluate(sample: int = 20, user=Depends(require_mod)):
# ------------------------------------------------------------------ # ------------------------------------------------------------------
# POST /api/admin/wiki/reset-gemma — Gemma-Einträge zurücksetzen # GET /api/admin/wiki/enrichment-status — Enrichment-Statistik
# ------------------------------------------------------------------ # ------------------------------------------------------------------
@router.post("/wiki/reset-gemma") @router.get("/wiki/enrichment-status")
async def wiki_reset_gemma(user=Depends(require_mod)): async def wiki_enrichment_status(user=Depends(require_mod)):
from scraper.breed_enricher import reset_gemma_entries with db() as conn:
count = reset_gemma_entries() total = conn.execute("SELECT COUNT(*) FROM wiki_rassen").fetchone()[0]
return {"reset": count} enriched = conn.execute("SELECT COUNT(*) FROM wiki_rassen WHERE ki_enriched=1").fetchone()[0]
no_wiki = conn.execute("SELECT COUNT(*) FROM wiki_rassen WHERE ki_enriched=2").fetchone()[0]
pending = conn.execute("SELECT COUNT(*) FROM wiki_rassen WHERE ki_enriched=0").fetchone()[0]
by_model = {
row[0] or "unbekannt": row[1]
for row in conn.execute(
"SELECT ki_model, COUNT(*) FROM wiki_rassen "
"WHERE ki_enriched=1 GROUP BY ki_model ORDER BY 2 DESC"
).fetchall()
}
with_photo = conn.execute(
"SELECT COUNT(*) FROM wiki_rassen WHERE foto_url IS NOT NULL AND foto_url != ''"
).fetchone()[0]
return {
"total": total,
"enriched": enriched,
"no_wiki": no_wiki,
"pending": pending,
"with_photo": with_photo,
"by_model": by_model,
}
# ------------------------------------------------------------------ # ------------------------------------------------------------------
# POST /api/admin/wiki/translate-temperament — einmalige Migration # POST /api/admin/wiki/fetch-photos — Wiki-Fotos laden
# ------------------------------------------------------------------ # ------------------------------------------------------------------
@router.post("/wiki/translate-temperament") @router.post("/wiki/fetch-photos")
async def wiki_translate_temperament(user=Depends(require_mod)): async def wiki_fetch_photos(limit: int = 50, user=Depends(require_mod)):
from scraper.breed_enricher import translate_existing_temperaments import asyncio, subprocess
updated = translate_existing_temperaments() proc = await asyncio.create_subprocess_exec(
return {"updated": updated} "python3", "/app/scraper/fetch_wiki_images.py", "--limit", str(limit),
stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE
)
stdout, stderr = await proc.communicate()
lines = (stdout + stderr).decode()
found = lines.count("Foto gespeichert")
return {"launched": True, "found": found, "log": lines[-2000:]}
# ------------------------------------------------------------------ # ------------------------------------------------------------------

View file

@ -3,7 +3,7 @@
Router, State-Management, Navigation, Initialisierung. Router, State-Management, Navigation, Initialisierung.
============================================================ */ ============================================================ */
const APP_VER = '345'; // ← bei jedem Deploy mit Frontend-Änderungen erhöhen const APP_VER = '346'; // ← bei jedem Deploy mit Frontend-Änderungen erhöhen
const App = (() => { const App = (() => {

View file

@ -739,14 +739,14 @@ window.Page_admin = (() => {
<div style="font-size:var(--text-sm);font-weight:var(--weight-semibold); <div style="font-size:var(--text-sm);font-weight:var(--weight-semibold);
color:var(--c-text);margin-bottom:var(--space-3)">Wartung</div> color:var(--c-text);margin-bottom:var(--space-3)">Wartung</div>
<div style="display:flex;flex-wrap:wrap;gap:var(--space-2)"> <div style="display:flex;flex-wrap:wrap;gap:var(--space-2)">
<button class="btn btn-secondary btn-sm" id="adm-translate-temper"> <button class="btn btn-secondary btn-sm" id="adm-enrichment-status">
${UI.icon('translate')} Temperament Deutsch ${UI.icon('arrows-clockwise')} Enrichment-Status
</button> </button>
<button class="btn btn-secondary btn-sm" id="adm-evaluate-breeds"> <button class="btn btn-secondary btn-sm" id="adm-evaluate-breeds">
${UI.icon('chart-bar')} Qualitätsbewertung (20 Rassen) ${UI.icon('chart-bar')} Qualitätsbewertung (20 Rassen)
</button> </button>
<button class="btn btn-danger btn-sm" id="adm-reset-gemma"> <button class="btn btn-secondary btn-sm" id="adm-fetch-photos">
${UI.icon('arrow-counter-clockwise')} Gemma-Einträge zurücksetzen ${UI.icon('image')} Fotos laden
</button> </button>
</div> </div>
<div id="adm-maint-result" style="margin-top:var(--space-2);font-size:var(--text-xs); <div id="adm-maint-result" style="margin-top:var(--space-2);font-size:var(--text-xs);
@ -792,14 +792,16 @@ window.Page_admin = (() => {
}); });
el.querySelector('#adm-log-refresh').addEventListener('click', loadLogs); el.querySelector('#adm-log-refresh').addEventListener('click', loadLogs);
el.querySelector('#adm-log-level').addEventListener('change', loadLogs); el.querySelector('#adm-log-level').addEventListener('change', loadLogs);
el.querySelector('#adm-translate-temper').addEventListener('click', async (e) => { el.querySelector('#adm-enrichment-status').addEventListener('click', async (e) => {
const btn = e.currentTarget; const btn = e.currentTarget;
const res = el.querySelector('#adm-maint-result'); const res = el.querySelector('#adm-maint-result');
btn.disabled = true; btn.disabled = true;
res.textContent = 'Läuft…'; res.textContent = 'Lade…';
try { try {
const d = await API.post('/admin/wiki/translate-temperament', {}); const d = await API.get('/admin/wiki/enrichment-status');
res.textContent = `${d.updated} Rassen übersetzt`; const modelList = Object.entries(d.by_model)
.map(([m, n]) => `${m}: ${n}`).join(', ');
res.textContent = `Gesamt: ${d.total} | Angereichert: ${d.enriched} | Kein Wiki: ${d.no_wiki} | Ausstehend: ${d.pending} | Mit Foto: ${d.with_photo} | Modelle: ${modelList || ''}`;
} catch (err) { } catch (err) {
res.textContent = '✗ Fehler: ' + (err.message || err); res.textContent = '✗ Fehler: ' + (err.message || err);
} finally { } finally {
@ -807,14 +809,14 @@ window.Page_admin = (() => {
} }
}); });
el.querySelector('#adm-reset-gemma').addEventListener('click', async (e) => { el.querySelector('#adm-fetch-photos').addEventListener('click', async (e) => {
if (!confirm('Alle Gemma-angereicherten Einträge zurücksetzen? Sie werden beim nächsten Job neu (Wikipedia-grounded) angereichert.')) return;
const btn = e.currentTarget; const btn = e.currentTarget;
const res = el.querySelector('#adm-maint-result'); const res = el.querySelector('#adm-maint-result');
btn.disabled = true; btn.disabled = true;
res.textContent = 'Fotos werden geladen… (kann 3060s dauern)';
try { try {
const d = await API.post('/admin/wiki/reset-gemma', {}); const d = await API.post('/admin/wiki/fetch-photos?limit=50', {});
res.textContent = `${d.reset} Gemma-Einträge zurückgesetzt`; res.textContent = `${d.found} Foto(s) gespeichert`;
} catch (err) { } catch (err) {
res.textContent = '✗ Fehler: ' + (err.message || err); res.textContent = '✗ Fehler: ' + (err.message || err);
} finally { } finally {

View file

@ -3,7 +3,7 @@
Offline-Cache + Push Notifications + Tile-Cache Offline-Cache + Push Notifications + Tile-Cache
============================================================ */ ============================================================ */
const CACHE_VERSION = 'by-v360'; const CACHE_VERSION = 'by-v361';
const CACHE_STATIC = `${CACHE_VERSION}-static`; const CACHE_STATIC = `${CACHE_VERSION}-static`;
const CACHE_TILES = 'ban-yaro-tiles-v1'; // bleibt über SW-Updates erhalten const CACHE_TILES = 'ban-yaro-tiles-v1'; // bleibt über SW-Updates erhalten