#!/usr/bin/env python3
import argparse, collections

def read_trec(path):
    byq = collections.defaultdict(list)
    with open(path, encoding="utf-8", errors="ignore") as f:
        for ln in f:
            parts = ln.strip().split()
            if len(parts) < 6: continue
            qid, _, docid, rank, score, tag = parts[:6]
            byq[qid].append((docid, int(rank), float(score), tag))
    for q in byq:
        byq[q].sort(key=lambda x: x[1])
    return byq

def main():
    ap = argparse.ArgumentParser()
    ap.add_argument("--run", required=True)
    ap.add_argument("--out", required=True)
    ap.add_argument("--K", type=float, default=60.0, help="WRRF K")
    ap.add_argument("--cutoff", type=int, default=100)
    args = ap.parse_args()

    runs = read_trec(args.run)
    with open(args.out, "w", encoding="utf-8") as w:
        for q, lst in runs.items():
            rescored = []
            for docid, r, _, _ in lst:
                s = 1.0 / (args.K + r)  # WRRF
                rescored.append((docid, r, s))
            # sort by score desc then original rank
            rescored.sort(key=lambda x: (-x[2], x[1]))
            for i,(d, r, s) in enumerate(rescored[:args.cutoff], 1):
                w.write(f"{q} Q0 {d} {i} {s:.6f} BGE_WRRF\n")

if __name__ == "__main__":
    main()
