from flask import Flask, request, redirect import os, json, requests from redis import Redis from rq import Queue MEILI_URL = os.getenv("MEILI_URL", "http://meili:7700") MEILI_KEY = os.getenv("MEILI_KEY", "") # from .env REDIS_URL = os.getenv("REDIS_URL", "redis://redis:6379/0") app = Flask(__name__) q = Queue(connection=Redis.from_url(REDIS_URL)) PAGE = """ PodX - unified search

PodX

Batch

Unified search (podcasts + PDFs + EPUB + Kiwix + Web)

Recent jobs

""" def meili_search(qstr, limit=30): if not qstr.strip(): return [] r = requests.post(f"{MEILI_URL}/indexes/library/search", headers={"Authorization": f"Bearer {MEILI_KEY}","Content-Type":"application/json"}, data=json.dumps({"q": qstr, "limit": limit})) return r.json().get("hits", []) @app.get("/") def index(): return PAGE @app.post("/enqueue") def enqueue(): url = request.form["url"].strip() q.enqueue("worker.handle_url", url) return redirect("/") @app.post("/enqueue_batch") def enqueue_batch(): urls = [u.strip() for u in request.form["urls"].splitlines() if u.strip()] for u in urls: q.enqueue("worker.handle_url", u) return redirect("/") @app.get("/recent") def recent(): try: with open("/transcripts/_feed.log", "r", encoding="utf-8") as f: tail = f.readlines()[-40:] except FileNotFoundError: tail=[] html = [] for line in reversed(tail): try: item = json.loads(line) except: continue html.append(f"
{item.get('title','')}
{item.get('uploader','')} — {item.get('date','')} — {item.get('status','')}
{item.get('path','')}
") return "\n".join(html) @app.get("/search") def search(): qstr = request.args.get("q","") hits = meili_search(qstr) out=[] for h in hits: t = h.get("title","") src = h.get("source","") typ = h.get("type","") ctx = h.get("_formatted",{}).get("text", h.get("text","")[:300]) segs = h.get("segments",[]) ts = int(segs[0]["start"]) if segs else 0 if typ == 'podcast': open_link = f"/open?file={requests.utils.quote(src)}&t={ts}" else: open_link = f"/open?file={requests.utils.quote(src)}" transcript_link = f" | Transcript" if typ == 'podcast' else "" badge = f"{typ}" out.append( f"
{badge}{t}
{src}" f"

{ctx}

" f"Open" f"{transcript_link}" f"
" ) return "\n".join(out) or "No results yet." @app.get("/open") def open_local(): file = request.args.get("file","") t = int(request.args.get("t","0")) return f"
{file}\nStart at: {t} sec
" @app.get("/subtitle") def subtitle(): file = request.args.get("file","") base = os.path.splitext(os.path.basename(file))[0] p = f"/transcripts/{base}.vtt" if os.path.exists(p): with open(p,"r",encoding="utf-8") as f: return f"
{f.read()}
" return "No VTT found."