{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "3984513a",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import re\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "3226b95e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import csv\n",
    "import logging\n",
    "from typing import Dict, List, Tuple\n",
    "\n",
    "import numpy as np\n",
    "import pytrec_eval\n",
    "\n",
    "def evaluate(\n",
    "    qrels: Dict[str, Dict[str, int]],\n",
    "    results: Dict[str, Dict[str, float]],\n",
    "    k_values: List[int],\n",
    ") -> Tuple[Dict[str, float], Dict[str, float], Dict[str, float], Dict[str, float]]:\n",
    "    \"\"\"\n",
    "    仿照 beir.retrieval.evaluation.EvaluateRetrieval.evaluate 编写的评估函数。\n",
    "    \"\"\"\n",
    "    ndcg = {}\n",
    "    _map = {}\n",
    "    recall = {}\n",
    "    precision = {}\n",
    "\n",
    "    for k in k_values:\n",
    "        ndcg[f\"NDCG@{k}\"] = 0.0\n",
    "        _map[f\"MAP@{k}\"] = 0.0\n",
    "        recall[f\"Recall@{k}\"] = 0.0\n",
    "        precision[f\"P@{k}\"] = 0.0\n",
    "\n",
    "    map_string = \"map_cut.\" + \",\".join([str(k) for k in k_values])\n",
    "    ndcg_string = \"ndcg_cut.\" + \",\".join([str(k) for k in k_values])\n",
    "    recall_string = \"recall.\" + \",\".join([str(k) for k in k_values])\n",
    "    precision_string = \"P.\" + \",\".join([str(k) for k in k_values])\n",
    "    evaluator = pytrec_eval.RelevanceEvaluator(\n",
    "        qrels, {map_string, ndcg_string, recall_string, precision_string}\n",
    "    )\n",
    "    scores = evaluator.evaluate(results)\n",
    "\n",
    "    for query_id in scores.keys():\n",
    "        for k in k_values:\n",
    "            ndcg[f\"NDCG@{k}\"] += scores[query_id][\"ndcg_cut_\" + str(k)]\n",
    "            _map[f\"MAP@{k}\"] += scores[query_id][\"map_cut_\" + str(k)]\n",
    "            recall[f\"Recall@{k}\"] += scores[query_id][\"recall_\" + str(k)]\n",
    "            precision[f\"P@{k}\"] += scores[query_id][\"P_\" + str(k)]\n",
    "\n",
    "    for k in k_values:\n",
    "        ndcg[f\"NDCG@{k}\"] = round(ndcg[f\"NDCG@{k}\"] / len(scores), 5)\n",
    "        _map[f\"MAP@{k}\"] = round(_map[f\"MAP@{k}\"] / len(scores), 5)\n",
    "        recall[f\"Recall@{k}\"] = round(recall[f\"Recall@{k}\"] / len(scores), 5)\n",
    "        precision[f\"P@{k}\"] = round(precision[f\"P@{k}\"] / len(scores), 5)\n",
    "\n",
    "    # for eval_metric in [ndcg, _map, recall, precision]:\n",
    "    #     logging.info(\"\\n\")\n",
    "    #     for k, v in eval_metric.items():\n",
    "    #         logging.info(f\"{k}: {v:.4f}\")\n",
    "\n",
    "    return recall\n",
    "\n",
    "\n",
    "def load_gt(gt_path: str) -> Dict[str, Dict[str, int]]:\n",
    "    \"\"\"\n",
    "    加载 ground-truth npy 文件并转换为 pytrec_eval 所需的 qrels 格式。\n",
    "    查询ID将使用其在npy文件中的索引（0, 1, 2, ...）。\n",
    "    \"\"\"\n",
    "    gt_data = np.load(gt_path, allow_pickle=True)\n",
    "    gt_data = gt_data.reshape(-1, 1)\n",
    "    # gt_data = gt_data[:1000]\n",
    "    qrels = {}\n",
    "    for i, gt_list in enumerate(gt_data):\n",
    "        query_id = str(i)\n",
    "        qrels[query_id] = {}\n",
    "        for passage_id in gt_list:\n",
    "            qrels[query_id][str(passage_id)] = 1  # 假设相关性得分为1\n",
    "    return qrels\n",
    "\n",
    "\n",
    "def load_results(results_path: str) -> Dict[str, Dict[str, float]]:\n",
    "    \"\"\"\n",
    "    加载检索结果的tsv文件并转换为 pytrec_eval 所需的 results 格式。\n",
    "    - 如果文件有4列 (query_id, passage_id, rank, score)，则使用第四列的分数。\n",
    "    - 如果文件只有3列 (query_id, passage_id, rank)，则使用 1/rank 作为分数。\n",
    "    \"\"\"\n",
    "    results = {}\n",
    "    with open(results_path, \"r\", encoding=\"utf-8\") as f:\n",
    "        reader = csv.reader(f, delimiter=\"\\t\")\n",
    "        for row in reader:\n",
    "            if not row: continue # Skip empty lines\n",
    "\n",
    "            query_id, passage_id = row[0], row[1]\n",
    "            \n",
    "            if query_id not in results:\n",
    "                results[query_id] = {}\n",
    "            \n",
    "            # 判断使用真实分数还是生成代理分数\n",
    "            if len(row) == 4:\n",
    "                score = float(row[3])\n",
    "            elif len(row) == 3:\n",
    "                rank = int(row[2])\n",
    "                score = 1.0 / rank\n",
    "            else:\n",
    "                logging.warning(f\"Skipping malformed line with {len(row)} columns: {row}\")\n",
    "                continue\n",
    "            \n",
    "            results[query_id][passage_id] = score\n",
    "            \n",
    "    return results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "e7ba3ccd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Found 5 files for dataset 'clerc_128-IGP':\n",
      "\n",
      "/data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_10.tsv\n",
      "/data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_100.tsv\n",
      "/data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_1000.tsv\n",
      "/data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_50.tsv\n",
      "/data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_500.tsv\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "\n",
    "# Base directory\n",
    "base_dir = \"/data/lijunlin/sigmod2025-results/rebuttal/answer\"\n",
    "\n",
    "# Dataset keyword to filter by (e.g., \"clip\", \"clerc\", \"multiqa_med\")\n",
    "dataset_keyword = \"clerc_128-IGP\"\n",
    "\n",
    "# Collect full paths for matching files\n",
    "matching_paths = [\n",
    "    os.path.join(base_dir, f)\n",
    "    for f in os.listdir(base_dir)\n",
    "    if f.startswith(dataset_keyword + \"-\") and f.endswith(\".tsv\")\n",
    "]\n",
    "\n",
    "# Print results\n",
    "if matching_paths:\n",
    "    print(f\"Found {len(matching_paths)} files for dataset '{dataset_keyword}':\\n\")\n",
    "    for path in sorted(matching_paths):\n",
    "        print(path)\n",
    "else:\n",
    "    print(f\"No files found for dataset '{dataset_keyword}'.\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "e842b785",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2026-01-07 11:08:04 - Loading ground-truth file /home/ali/hnswlib/evaluation/clerc_gt.npy...\n",
      "2026-01-07 11:08:04 - Evaluating results file: /data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_10.tsv ...\n",
      "2026-01-07 11:08:05 - ✅ Finished evaluating clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_10.tsv\n",
      "2026-01-07 11:08:05 - Evaluating results file: /data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_100.tsv ...\n",
      "2026-01-07 11:08:05 - ✅ Finished evaluating clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_100.tsv\n",
      "2026-01-07 11:08:05 - Evaluating results file: /data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_1000.tsv ...\n",
      "2026-01-07 11:08:05 - ✅ Finished evaluating clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_1000.tsv\n",
      "2026-01-07 11:08:05 - Evaluating results file: /data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_50.tsv ...\n",
      "2026-01-07 11:08:06 - ✅ Finished evaluating clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_50.tsv\n",
      "2026-01-07 11:08:06 - Evaluating results file: /data/lijunlin/sigmod2025-results/rebuttal/answer/clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_500.tsv ...\n",
      "2026-01-07 11:08:06 - ✅ Finished evaluating clerc_128-IGP-top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_500.tsv\n",
      "2026-01-07 11:08:06 - 🎯 All evaluations completed.\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import logging\n",
    "\n",
    "# --- Logging setup ---\n",
    "logging.basicConfig(\n",
    "    format=\"%(asctime)s - %(message)s\",\n",
    "    datefmt=\"%Y-%m-%d %H:%M:%S\",\n",
    "    level=logging.INFO,\n",
    ")\n",
    "\n",
    "# --- Ground-truth file ---\n",
    "GT_FILE_PATH = \"/home/ali/hnswlib/evaluation/clerc_gt.npy\"\n",
    "\n",
    "# --- Base results folder ---\n",
    "BASE_RESULTS_DIR = \"/data/lijunlin/sigmod2025-results/rebuttal/answer\"\n",
    "\n",
    "# --- Dataset filter ---\n",
    "DATASET_KEYWORD = \"clerc_128-IGP\"  # Change to e.g. \"clerc\", \"multiqa_med\", etc.\n",
    "\n",
    "# --- Evaluation parameters ---\n",
    "K_VALUES = [10]\n",
    "\n",
    "# --- Your existing functions (assumed imported) ---\n",
    "# from your_module import load_gt, load_results, evaluate\n",
    "\n",
    "# --- Step 1: Collect all matching result files ---\n",
    "result_paths = [\n",
    "    os.path.join(BASE_RESULTS_DIR, f)\n",
    "    for f in os.listdir(BASE_RESULTS_DIR)\n",
    "    if f.startswith(DATASET_KEYWORD + \"-\") and f.endswith(\".tsv\")\n",
    "]\n",
    "\n",
    "if not result_paths:\n",
    "    logging.warning(f\"No result files found for dataset '{DATASET_KEYWORD}'.\")\n",
    "    exit()\n",
    "\n",
    "# --- Step 2: Load GT once ---\n",
    "logging.info(f\"Loading ground-truth file {GT_FILE_PATH}...\")\n",
    "qrels_data = load_gt(GT_FILE_PATH)\n",
    "\n",
    "# --- Step 3: Iterate over all result files ---\n",
    "for path in sorted(result_paths):\n",
    "    logging.info(f\"Evaluating results file: {path} ...\")\n",
    "    try:\n",
    "        results_data = load_results(path)\n",
    "        evaluate(qrels=qrels_data, results=results_data, k_values=K_VALUES)\n",
    "        logging.info(f\"✅ Finished evaluating {os.path.basename(path)}\")\n",
    "    except Exception as e:\n",
    "        logging.error(f\"❌ Failed to evaluate {path}: {e}\")\n",
    "\n",
    "logging.info(\"🎯 All evaluations completed.\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "71f13dfe",
   "metadata": {},
   "outputs": [],
   "source": [
    "algorithms = [\"IGP\", \"Dessert\", \"Plaid\", \"MUVERA\", \"Multi-HNSW\", \"HNSW-D\", \"C-Multi-HNSW\"]\n",
    "datasets = [\"CLERC-128\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "a3cdff95",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
      "/tmp/ipykernel_155498/2343904521.py:76: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Algorithm</th>\n",
       "      <th>QPS</th>\n",
       "      <th>Dataset</th>\n",
       "      <th>Recall</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>0.237029</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.131</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>0.272267</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.054</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>0.836701</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.084</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>0.387580</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.139</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>1.120122</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.050</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>0.832868</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.069</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>1.739312</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.028</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>HNSW-D</td>\n",
       "      <td>0.593912</td>\n",
       "      <td>CLERC-128</td>\n",
       "      <td>0.026</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "  Algorithm       QPS    Dataset  Recall\n",
       "0    HNSW-D  0.237029  CLERC-128   0.131\n",
       "1    HNSW-D  0.272267  CLERC-128   0.054\n",
       "2    HNSW-D  0.836701  CLERC-128   0.084\n",
       "3    HNSW-D  0.387580  CLERC-128   0.139\n",
       "4    HNSW-D  1.120122  CLERC-128   0.050\n",
       "5    HNSW-D  0.832868  CLERC-128   0.069\n",
       "6    HNSW-D  1.739312  CLERC-128   0.028\n",
       "7    HNSW-D  0.593912  CLERC-128   0.026"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import os\n",
    "import re\n",
    "import pandas as pd\n",
    "\n",
    "# REQUIRED: you should already have these defined in your notebook\n",
    "# - qrels_data  (from your load_gt)\n",
    "# - K_VALUES    (e.g., [10])\n",
    "# - evaluate(qrels, results, k_values) -> dict of metrics (e.g., {'Recall@10': 0.736})\n",
    "\n",
    "# ----------------------------\n",
    "# Config\n",
    "# ----------------------------\n",
    "directory = \"/data/lijunlin/sigmod2025-results/rebuttal/multi-hnsw-result\"\n",
    "# keyword = \"clip-multi-clustering_HNSW\"\n",
    "keyword = \"clerc_128_HNSW\"\n",
    "# keyword = \"clef_med_HNSW\"\n",
    "# keyword = \"multiqa_med_HNSW\"\n",
    "\n",
    "# ----------------------------\n",
    "# File pairing\n",
    "# ----------------------------\n",
    "all_files = os.listdir(directory)\n",
    "txt_files = [\n",
    "    f for f in all_files\n",
    "    if f.endswith(\".txt\") and keyword in f and \"metadata\" not in f\n",
    "]\n",
    "\n",
    "file_pairs = {}\n",
    "for f in txt_files:\n",
    "    if f.endswith(\"_summary.txt\"):\n",
    "        base_name = f[:-len(\"_summary.txt\")]\n",
    "        file_pairs.setdefault(base_name, {})[\"summary\"] = os.path.join(directory, f)\n",
    "    else:\n",
    "        base_name = f[:-len(\".txt\")]\n",
    "        file_pairs.setdefault(base_name, {})[\"results\"] = os.path.join(directory, f)\n",
    "\n",
    "# ----------------------------\n",
    "# Helpers\n",
    "# ----------------------------\n",
    "def load_results_tsv(path: str) -> pd.DataFrame:\n",
    "    # Results: query_id, passage_id, rank, score (tab-separated, no header)\n",
    "    return pd.read_csv(\n",
    "        path,\n",
    "        sep=\"\\t\",\n",
    "        header=None,\n",
    "        names=[\"query_id\", \"passage_id\", \"rank\", \"score\"],\n",
    "        dtype={\"query_id\": int, \"passage_id\": int, \"rank\": int, \"score\": float}\n",
    "    )\n",
    "\n",
    "def extract_avg_retrieval_time(summary_text: str) -> float | None:\n",
    "    # Robust parser for avg retrieval time (ms)\n",
    "    patterns = [\n",
    "        r\"Average\\s+Query\\s+Time:\\s*([\\d.]+)\\s*ms\",\n",
    "        r\"Avg\\s*retrieval\\s*time\\s*[:=]\\s*([\\d.]+)\\s*ms\",\n",
    "        r\"retrieval_time_single_query_average\\(ms\\)\\s*[:=]\\s*([\\d.]+)\",\n",
    "        r\"avg.*?ms\\s*[:=]\\s*([\\d.]+)\",\n",
    "    ]\n",
    "    for pat in patterns:\n",
    "        m = re.search(pat, summary_text, flags=re.IGNORECASE)\n",
    "        if m:\n",
    "            try:\n",
    "                return float(m.group(1))\n",
    "            except ValueError:\n",
    "                pass\n",
    "    return None\n",
    "\n",
    "def df_results_to_dict(df_res: pd.DataFrame) -> dict:\n",
    "    # Drop duplicates (query, passage) keeping highest score; cast IDs to str\n",
    "    df = (\n",
    "        df_res.sort_values('score', ascending=False)\n",
    "              .drop_duplicates(['query_id', 'passage_id'])\n",
    "              .astype({'query_id': str, 'passage_id': str})\n",
    "    )\n",
    "    return (\n",
    "        df.groupby('query_id')\n",
    "          .apply(lambda g: dict(zip(g['passage_id'], g['score'])))\n",
    "          .to_dict()\n",
    "    )\n",
    "\n",
    "# ----------------------------\n",
    "# Build final DataFrame: one row per base with qps + metrics\n",
    "# ----------------------------\n",
    "rows = []\n",
    "for base, pair in file_pairs.items():\n",
    "    # Summary → avg_ms → qps\n",
    "    avg_ms = None\n",
    "    if 'summary' in pair and pair['summary']:\n",
    "        with open(pair['summary'], 'r', encoding='utf-8', errors='ignore') as f:\n",
    "            avg_ms = extract_avg_retrieval_time(f.read())\n",
    "    qps = (1000.0 / avg_ms) if (avg_ms is not None and avg_ms > 0) else None\n",
    "\n",
    "    # Results → metrics via evaluate\n",
    "    metrics = {}\n",
    "    if 'results' in pair and pair['results']:\n",
    "        df_res = load_results_tsv(pair['results'])\n",
    "        results_dict = df_results_to_dict(df_res)\n",
    "        metrics = evaluate(qrels=qrels_data, results=results_dict, k_values=K_VALUES) or {}\n",
    "\n",
    "    row = {'Algorithm': algorithms[5], 'QPS': qps, \"Dataset\": datasets[0], \"Recall\": metrics[\"Recall@10\"]}\n",
    "    # row.update(metrics)  # adds e.g. 'Recall@10', etc.\n",
    "    rows.append(row)\n",
    "\n",
    "mvrhnsw_df = pd.DataFrame(rows).reset_index(drop=True)\n",
    "mvrhnsw_df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d129a374",
   "metadata": {},
   "outputs": [],
   "source": [
    "mvrhnsw_df.to_csv(\"/home/ali/hnswlib/evaluation/results_MVRHNSW.csv\", index=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "c111e196",
   "metadata": {},
   "outputs": [],
   "source": [
    "def extract_limit(row):\n",
    "    identifier = row['Identifier']\n",
    "    name = row['Algorithm']\n",
    "    \n",
    "    # Default to None\n",
    "    match = None\n",
    "\n",
    "    if name == \"IGP\":\n",
    "        # Extract probe_topk\n",
    "        match = re.search(r'probe_topk_(\\d+)', identifier)\n",
    "    \n",
    "    if match:\n",
    "        return int(match.group(1))\n",
    "    else:\n",
    "        return 0  # If pattern not found, assume safe to keep\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "635d46e9",
   "metadata": {},
   "outputs": [],
   "source": [
    "# --- Configuration ---\n",
    "JSON_ROOT_DIR = \"/data/lijunlin/sigmod2025-results/rebuttal/performance\"\n",
    "TSV_ROOT_DIR = \"/data/lijunlin/sigmod2025-results/rebuttal/answer/\"\n",
    "KEYWORD_JSON = \"clerc_128-retrieval-IGP\"\n",
    "KEYWORD_TSV = \"clerc_128-IGP\"\n",
    "\n",
    "# --- Find files with keyword ---\n",
    "def find_files_with_keyword(root_dir, keyword, ext):\n",
    "    matched_files = []\n",
    "    for dirpath, _, filenames in os.walk(root_dir):\n",
    "        for filename in filenames:\n",
    "            if filename.endswith(ext) and keyword in filename:\n",
    "                matched_files.append(os.path.join(dirpath, filename))\n",
    "    return matched_files\n",
    "\n",
    "# --- Load JSON file safely ---\n",
    "def load_json_file(filepath):\n",
    "    try:\n",
    "        with open(filepath, \"r\") as f:\n",
    "            return json.load(f)\n",
    "    except Exception as e:\n",
    "        logging.error(f\"Failed to read {filepath}: {e}\")\n",
    "        return None\n",
    "\n",
    "# --- Extract identifier from filename ---\n",
    "def get_identifier(filename, keyword):\n",
    "    base = os.path.basename(filename)\n",
    "    return base.replace(keyword, \"\").replace(\".json\", \"\").replace(\".tsv\", \"\")\n",
    "\n",
    "# --- Pair JSON and TSV files by shared identifier ---\n",
    "def pair_json_tsv(json_files, tsv_files, keyword_json, keyword_tsv):\n",
    "    json_map = {get_identifier(f, keyword_json): f for f in json_files}\n",
    "    tsv_map = {get_identifier(f, keyword_tsv): f for f in tsv_files}\n",
    "\n",
    "    pairs = []\n",
    "    for identifier, json_path in json_map.items():\n",
    "        if identifier in tsv_map:\n",
    "            pairs.append((identifier, json_path, tsv_map[identifier]))\n",
    "    return pairs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "bda3a584",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2025-12-07 09:11:27 - Processing keyword pair: JSON='clerc_128-retrieval-dessert', TSV='clerc_128-dessert' (Name='Dessert')\n",
      "2025-12-07 09:11:27 - Processing pair: -top10-n_table_64-initial_filter_k_50-nprobe_query_2-remove_centroid_dupes_True-n_thread_1\n",
      "2025-12-07 09:11:28 - Processing pair: -top10-n_table_64-initial_filter_k_500-nprobe_query_2-remove_centroid_dupes_True-n_thread_1\n",
      "2025-12-07 09:11:28 - Processing pair: -top10-n_table_64-initial_filter_k_1000-nprobe_query_2-remove_centroid_dupes_True-n_thread_1\n",
      "2025-12-07 09:11:28 - Processing pair: -top10-n_table_64-initial_filter_k_10-nprobe_query_2-remove_centroid_dupes_True-n_thread_1\n",
      "2025-12-07 09:11:28 - Processing pair: -top10-n_table_64-initial_filter_k_100-nprobe_query_2-remove_centroid_dupes_True-n_thread_1\n",
      "2025-12-07 09:11:28 - Processing keyword pair: JSON='clerc_128-retrieval-plaid', TSV='clerc_128-plaid' (Name='Plaid')\n",
      "2025-12-07 09:11:28 - Processing pair: -top10--ndocs_1000-ncells_2-centroid_score_threshold_0.50-n_thread_1\n",
      "2025-12-07 09:11:29 - Processing pair: -top10--ndocs_50-ncells_2-centroid_score_threshold_0.50-n_thread_1\n",
      "2025-12-07 09:11:29 - Processing pair: -top10--ndocs_10-ncells_2-centroid_score_threshold_0.50-n_thread_1\n",
      "2025-12-07 09:11:29 - Processing pair: -top10--ndocs_100-ncells_2-centroid_score_threshold_0.50-n_thread_1\n",
      "2025-12-07 09:11:29 - Processing pair: -top10--ndocs_200-ncells_2-centroid_score_threshold_0.50-n_thread_1\n",
      "2025-12-07 09:11:29 - Processing pair: -top10--ndocs_500-ncells_2-centroid_score_threshold_0.50-n_thread_1\n",
      "2025-12-07 09:11:30 - Processing keyword pair: JSON='clerc_128-retrieval-IGP', TSV='clerc_128-IGP' (Name='IGP')\n",
      "2025-12-07 09:11:30 - Processing pair: -top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_50\n",
      "2025-12-07 09:11:30 - Processing pair: -top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_500\n",
      "2025-12-07 09:11:30 - Processing pair: -top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_10\n",
      "2025-12-07 09:11:30 - Processing pair: -top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_1000\n",
      "2025-12-07 09:11:30 - Processing pair: -top10-n_centroid_1024-n_bit_2-nprobe_32-probe_topk_100\n",
      "/tmp/ipykernel_26120/2375339678.py:81: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  df_sorted = df_filtered.groupby(\"Algorithm\", group_keys=False).apply(\n"
     ]
    }
   ],
   "source": [
    "# --- Define multiple keyword pairs with names ---\n",
    "\n",
    "import json\n",
    "keyword_pairs = [\n",
    "    # (\"clip-multi-clustering-retrieval-IGP\", \"clip-multi-clustering-IGP\", \"IGP\"),\n",
    "    # (\"clip-multi-clustering-retrieval-dessert\", \"clip-multi-clustering-dessert\", \"Dessert\"),\n",
    "    # (\"clip-multi-clustering-retrieval-plaid\", \"clip-multi-clustering-plaid\", \"Plaid\"),\n",
    "\n",
    "    (\"clerc_128-retrieval-dessert\", \"clerc_128-dessert\",  \"Dessert\"),\n",
    "    (\"clerc_128-retrieval-plaid\",\"clerc_128-plaid\",  \"Plaid\"),\n",
    "    (\"clerc_128-retrieval-IGP\", \"clerc_128-IGP\", \"IGP\"),\n",
    "\n",
    "    # (\"clef_med-retrieval-dessert\", \"clef_med-dessert\",  \"Dessert\"),\n",
    "    # (\"clef_med-retrieval-plaid\",\"clef_med-plaid\",  \"Plaid\"),\n",
    "    # (\"clef_med-retrieval-IGP\", \"clef_med-IGP\", \"IGP\"),\n",
    "\n",
    "    # (\"multiqa_med-retrieval-dessert\", \"multiqa_med-dessert\",  \"Dessert\"),\n",
    "    # (\"multiqa_med-retrieval-plaid\",\"multiqa_med-plaid\",  \"Plaid\"),\n",
    "    # (\"multiqa_med-retrieval-IGP\", \"multiqa_med-IGP\", \"IGP\"),\n",
    "\n",
    "    \n",
    "    \n",
    "]\n",
    "\n",
    "all_results = []\n",
    "\n",
    "for KEYWORD_JSON, KEYWORD_TSV, name in keyword_pairs:\n",
    "\n",
    "    logging.info(f\"Processing keyword pair: JSON='{KEYWORD_JSON}', TSV='{KEYWORD_TSV}' (Name='{name}')\")\n",
    "    \n",
    "    # Find files\n",
    "    json_files = find_files_with_keyword(JSON_ROOT_DIR, KEYWORD_JSON, \".json\")\n",
    "    tsv_files = [\n",
    "        os.path.join(TSV_ROOT_DIR, f)\n",
    "        for f in os.listdir(TSV_ROOT_DIR)\n",
    "        if f.startswith(KEYWORD_TSV + \"-\") and f.endswith(\".tsv\")\n",
    "    ]\n",
    "\n",
    "    # Pair files by identifier\n",
    "    pairs = pair_json_tsv(json_files, tsv_files, KEYWORD_JSON, KEYWORD_TSV)\n",
    "\n",
    "    # Process each pair\n",
    "    for identifier, json_file, tsv_file in pairs:\n",
    "        logging.info(f\"Processing pair: {identifier}\")\n",
    "\n",
    "        data = load_json_file(json_file)\n",
    "        if data is None:\n",
    "            continue\n",
    "        \n",
    "        if name == 'IGP':\n",
    "            avg_ms = float(data['search_time']['retrieval_time_single_query_average(ms)'])\n",
    "        else:\n",
    "            avg_ms = float(data['search_time']['average_query_time_ms'])\n",
    "        qps = 1000 / avg_ms\n",
    "\n",
    "        results_data = load_results(tsv_file)\n",
    "        recall = evaluate(qrels=qrels_data, results=results_data, k_values=K_VALUES)\n",
    "\n",
    "        all_results.append({\n",
    "            \"Identifier\":identifier,\n",
    "            'Recall': recall['Recall@10'],\n",
    "            'QPS': qps,\n",
    "            'Algorithm': name,  # <-- Add name here\n",
    "            \"Dataset\": datasets[0]\n",
    "        })\n",
    "\n",
    "# Convert to DataFrame\n",
    "df_all_results = pd.DataFrame(all_results)\n",
    "\n",
    "# df_all_results['limit_value'] = df_all_results.apply(extract_limit, axis=1)\n",
    "\n",
    "# # Filter rows where limit_value <= 10000\n",
    "# df_filtered = df_all_results[df_all_results['limit_value'] <= 200000]\n",
    "\n",
    "\n",
    "# # Drop the temporary column\n",
    "# df_filtered.drop(columns=['limit_value'], inplace=True)\n",
    "df_filtered = df_all_results.copy()\n",
    "\n",
    "\n",
    "df_sorted = df_filtered.groupby(\"Algorithm\", group_keys=False).apply(\n",
    "    lambda x: x.sort_values(by=\"Recall\", ascending=True)\n",
    ")\n",
    "\n",
    "df_filtered =df_sorted"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "721d5ccb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Identifier</th>\n",
       "      <th>Recall</th>\n",
       "      <th>QPS</th>\n",
       "      <th>Algorithm</th>\n",
       "      <th>Dataset</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_50-nprobe_q...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>67.395877</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_500-nprobe_...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>26.883359</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_100-nprobe_...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>57.454516</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_1000-nprobe...</td>\n",
       "      <td>0.001</td>\n",
       "      <td>16.746572</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_10-nprobe_q...</td>\n",
       "      <td>0.001</td>\n",
       "      <td>77.159392</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>13</th>\n",
       "      <td>-top10-n_centroid_1024-n_bit_2-nprobe_32-probe...</td>\n",
       "      <td>0.008</td>\n",
       "      <td>1.198509</td>\n",
       "      <td>IGP</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>11</th>\n",
       "      <td>-top10-n_centroid_1024-n_bit_2-nprobe_32-probe...</td>\n",
       "      <td>0.019</td>\n",
       "      <td>1.201136</td>\n",
       "      <td>IGP</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>15</th>\n",
       "      <td>-top10-n_centroid_1024-n_bit_2-nprobe_32-probe...</td>\n",
       "      <td>0.023</td>\n",
       "      <td>1.184692</td>\n",
       "      <td>IGP</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>-top10-n_centroid_1024-n_bit_2-nprobe_32-probe...</td>\n",
       "      <td>0.033</td>\n",
       "      <td>1.120897</td>\n",
       "      <td>IGP</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>14</th>\n",
       "      <td>-top10-n_centroid_1024-n_bit_2-nprobe_32-probe...</td>\n",
       "      <td>0.040</td>\n",
       "      <td>1.055876</td>\n",
       "      <td>IGP</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>-top10--ndocs_10-ncells_2-centroid_score_thres...</td>\n",
       "      <td>0.001</td>\n",
       "      <td>1.916377</td>\n",
       "      <td>Plaid</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>-top10--ndocs_50-ncells_2-centroid_score_thres...</td>\n",
       "      <td>0.002</td>\n",
       "      <td>1.948816</td>\n",
       "      <td>Plaid</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>-top10--ndocs_100-ncells_2-centroid_score_thre...</td>\n",
       "      <td>0.002</td>\n",
       "      <td>1.951958</td>\n",
       "      <td>Plaid</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>-top10--ndocs_200-ncells_2-centroid_score_thre...</td>\n",
       "      <td>0.002</td>\n",
       "      <td>2.037220</td>\n",
       "      <td>Plaid</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>-top10--ndocs_500-ncells_2-centroid_score_thre...</td>\n",
       "      <td>0.005</td>\n",
       "      <td>1.975761</td>\n",
       "      <td>Plaid</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>-top10--ndocs_1000-ncells_2-centroid_score_thr...</td>\n",
       "      <td>0.009</td>\n",
       "      <td>1.975192</td>\n",
       "      <td>Plaid</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                           Identifier  Recall        QPS  \\\n",
       "0   -top10-n_table_64-initial_filter_k_50-nprobe_q...   0.000  67.395877   \n",
       "1   -top10-n_table_64-initial_filter_k_500-nprobe_...   0.000  26.883359   \n",
       "4   -top10-n_table_64-initial_filter_k_100-nprobe_...   0.000  57.454516   \n",
       "2   -top10-n_table_64-initial_filter_k_1000-nprobe...   0.001  16.746572   \n",
       "3   -top10-n_table_64-initial_filter_k_10-nprobe_q...   0.001  77.159392   \n",
       "13  -top10-n_centroid_1024-n_bit_2-nprobe_32-probe...   0.008   1.198509   \n",
       "11  -top10-n_centroid_1024-n_bit_2-nprobe_32-probe...   0.019   1.201136   \n",
       "15  -top10-n_centroid_1024-n_bit_2-nprobe_32-probe...   0.023   1.184692   \n",
       "12  -top10-n_centroid_1024-n_bit_2-nprobe_32-probe...   0.033   1.120897   \n",
       "14  -top10-n_centroid_1024-n_bit_2-nprobe_32-probe...   0.040   1.055876   \n",
       "7   -top10--ndocs_10-ncells_2-centroid_score_thres...   0.001   1.916377   \n",
       "6   -top10--ndocs_50-ncells_2-centroid_score_thres...   0.002   1.948816   \n",
       "8   -top10--ndocs_100-ncells_2-centroid_score_thre...   0.002   1.951958   \n",
       "9   -top10--ndocs_200-ncells_2-centroid_score_thre...   0.002   2.037220   \n",
       "10  -top10--ndocs_500-ncells_2-centroid_score_thre...   0.005   1.975761   \n",
       "5   -top10--ndocs_1000-ncells_2-centroid_score_thr...   0.009   1.975192   \n",
       "\n",
       "   Algorithm    Dataset  \n",
       "0    Dessert  CLERC-128  \n",
       "1    Dessert  CLERC-128  \n",
       "4    Dessert  CLERC-128  \n",
       "2    Dessert  CLERC-128  \n",
       "3    Dessert  CLERC-128  \n",
       "13       IGP  CLERC-128  \n",
       "11       IGP  CLERC-128  \n",
       "15       IGP  CLERC-128  \n",
       "12       IGP  CLERC-128  \n",
       "14       IGP  CLERC-128  \n",
       "7      Plaid  CLERC-128  \n",
       "6      Plaid  CLERC-128  \n",
       "8      Plaid  CLERC-128  \n",
       "9      Plaid  CLERC-128  \n",
       "10     Plaid  CLERC-128  \n",
       "5      Plaid  CLERC-128  "
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_sorted"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fed41e3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# my_sorted_recall_list = [0.845,0.862,0.876,0.874,0.877,0.874,0.875,0.856,0.857,0.867,0.88]\n",
    "# my_sorted_qps_list = [774.494,745.388,719.843,695.595,719.843,695.595,646.601,566.891,489.971,422.421,334.579]\n",
    "# df_temp = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'Multi-HNSW',\n",
    "#     'Dataset': datasets[0]\n",
    "# })\n",
    "# # my_sorted_recall_list = [0.259,0.4,0.472,0.533,0.569,0.667,0.719,0.742,0.773,0.769,0.772]\n",
    "# # my_sorted_qps_list = [11179.2,8459.66,7024,7007.71,6994.57,4917.48,4624.73,3934.13,3012.67,3141.93,3082.59]\n",
    "# # df_temp2 = pd.DataFrame({\n",
    "# #     'Recall': my_sorted_recall_list,\n",
    "# #     'QPS': my_sorted_qps_list,\n",
    "# #     'Algorithm': 'SVR-HNSW',\n",
    "# #     'Dataset': 'DBpedia-entity'\n",
    "# # })\n",
    "\n",
    "# my_sorted_recall_list = [0.883,\n",
    "# 0.883,\n",
    "# 0.883,\n",
    "# 0.883,\n",
    "# 0.883,\n",
    "# 0.883,\n",
    "# 0.894,\n",
    "# 0.903,\n",
    "# 0.907,\n",
    "# 0.906,\n",
    "# 0.906]\n",
    "# my_sorted_qps_list = [381.605,\n",
    "# 384.45,\n",
    "# 380.318,\n",
    "# 373.251,\n",
    "# 377.382,\n",
    "# 378.154,\n",
    "# 311.384,\n",
    "# 265.433,\n",
    "# 213.847,\n",
    "# 178.324,\n",
    "# 147.744]\n",
    "# df_temp3 = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'C-Multi-HNSW',\n",
    "#     'Dataset': datasets[0]\n",
    "# })\n",
    "\n",
    "# df_combined = pd.concat([df_sorted, df_temp], ignore_index=True)\n",
    "# df_combined = pd.concat([df_combined, df_temp3], ignore_index=True)\n",
    "\n",
    "# df_combined = df_combined.drop(columns=['Identifier'])\n",
    "# df_combined = pd.concat([df_combined, mvrhnsw_df], ignore_index=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6367cf7c",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_sorted_recall_list = [0.171098,\n",
    "0.20851,\n",
    "0.219642,\n",
    "0.223158,\n",
    "0.225467,\n",
    "0.225192,\n",
    "0.225949,\n",
    "0.225803,\n",
    "0.224731,\n",
    "0.223237,\n",
    "0.222254]\n",
    "my_sorted_qps_list = [\n",
    "1428.37,\n",
    "1316.82,\n",
    "1246.59,\n",
    "1189.44,\n",
    "1128.77,\n",
    "912.563,\n",
    "768.442,\n",
    "654.318,\n",
    "548.927,\n",
    "472.615,\n",
    "421.388,\n",
    "]\n",
    "df_temp = pd.DataFrame({\n",
    "    'Recall': my_sorted_recall_list,\n",
    "    'QPS': my_sorted_qps_list,\n",
    "    'Algorithm': 'Multi-HNSW',\n",
    "    'Dataset': datasets[0]\n",
    "})\n",
    "\n",
    "\n",
    "my_sorted_recall_list = [0.195,\n",
    "0.202,\n",
    "0.205,\n",
    "0.206,\n",
    "0.205,\n",
    "0.202,\n",
    "0.208,\n",
    "0.205,\n",
    "0.198,\n",
    "0.188]\n",
    "my_sorted_qps_list = [1168.44,\n",
    "1089.73,\n",
    "1016.52,\n",
    "960.88,\n",
    "904.67,\n",
    "749.31,\n",
    "628.95,\n",
    "533.12,\n",
    "454.79,\n",
    "398.24]\n",
    "df_temp3 = pd.DataFrame({\n",
    "    'Recall': my_sorted_recall_list,\n",
    "    'QPS': my_sorted_qps_list,\n",
    "    'Algorithm': 'C-Multi-HNSW',\n",
    "    'Dataset': datasets[0]\n",
    "})\n",
    "\n",
    "\n",
    "# my_sorted_recall_list = [0.259,0.4,0.472,0.533,0.569,0.667,0.719,0.742,0.773,0.769,0.772]\n",
    "# my_sorted_qps_list = [11179.2,8459.66,7024,7007.71,6994.57,4917.48,4624.73,3934.13,3012.67,3141.93,3082.59]\n",
    "# df_temp2 = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'SVR-HNSW',\n",
    "#     'Dataset': 'DBpedia-entity'\n",
    "# })\n",
    "\n",
    "df_combined = pd.concat([df_sorted, df_temp], ignore_index=True)\n",
    "# df_combined = pd.concat([df_combined, df_temp2], ignore_index=True)\n",
    "df_combined = pd.concat([df_combined, df_temp3], ignore_index=True)\n",
    "# df_combined = df_combined.drop(columns=['Identifier'])\n",
    "# df_combined = pd.concat([df_combined], ignore_index=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ffa4077c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# my_sorted_recall_list = [0.094334,\n",
    "# 0.106585,\n",
    "# 0.108116,\n",
    "# 0.110873,\n",
    "# 0.108729,\n",
    "# 0.106585,\n",
    "# 0.107198,\n",
    "# 0.10536,\n",
    "# 0.103522,\n",
    "# 0.105054,\n",
    "# 0.104441]\n",
    "# my_sorted_qps_list = [254.493,\n",
    "# 217.65,\n",
    "# 191.319,\n",
    "# 171.775,\n",
    "# 157.915,\n",
    "# 111.928,\n",
    "# 84.1819,\n",
    "# 68.034,\n",
    "# 49.712,\n",
    "# 39.3898,\n",
    "# 32.676]\n",
    "# df_temp = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'Multi-HNSW',\n",
    "#     'Dataset': datasets[2]\n",
    "# })\n",
    "\n",
    "# my_sorted_recall_list = [0.107198,\n",
    "# 0.108423,\n",
    "# 0.108116,\n",
    "# 0.108116,\n",
    "# 0.10781,\n",
    "# 0.10781,\n",
    "# 0.10536,\n",
    "# 0.105972,\n",
    "# 0.104747,\n",
    "# 0.105972,\n",
    "# 0.105972]\n",
    "# my_sorted_qps_list = [ 109.379,\n",
    "# 109.121,\n",
    "# 108.866,\n",
    "# 109.753,\n",
    "# 109.11,\n",
    "# 109.069,\n",
    "# 83.1861,\n",
    "# 68.3078,\n",
    "# 50.4555,\n",
    "# 40.2429,\n",
    "# 33.6657]\n",
    "# df_temp3 = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'C-Multi-HNSW',\n",
    "#     'Dataset': datasets[2]\n",
    "# })\n",
    "# # my_sorted_recall_list = [0.259,0.4,0.472,0.533,0.569,0.667,0.719,0.742,0.773,0.769,0.772]\n",
    "# # my_sorted_qps_list = [11179.2,8459.66,7024,7007.71,6994.57,4917.48,4624.73,3934.13,3012.67,3141.93,3082.59]\n",
    "# # df_temp2 = pd.DataFrame({\n",
    "# #     'Recall': my_sorted_recall_list,\n",
    "# #     'QPS': my_sorted_qps_list,\n",
    "# #     'Algorithm': 'SVR-HNSW',\n",
    "# #     'Dataset': 'DBpedia-entity'\n",
    "# # })\n",
    "\n",
    "# df_combined = pd.concat([df_sorted, df_temp], ignore_index=True)\n",
    "# df_combined = pd.concat([df_combined, df_temp3], ignore_index=True)\n",
    "\n",
    "# df_combined = df_combined.drop(columns=['Identifier'])\n",
    "# df_combined = pd.concat([df_combined, mvrhnsw_df], ignore_index=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "df47a660",
   "metadata": {},
   "outputs": [],
   "source": [
    "# my_sorted_recall_list = [0.332871,\n",
    "# 0.446602,\n",
    "# 0.493759,\n",
    "# 0.542302,\n",
    "# 0.568655,\n",
    "# 0.619972,\n",
    "# 0.654646,\n",
    "# 0.665742,\n",
    "# 0.690707,\n",
    "# 0.704577,\n",
    "# 0.712899]\n",
    "# my_sorted_qps_list = [3011.38,\n",
    "# 2994.4,\n",
    "# 2988.48,\n",
    "# 2958.11,\n",
    "# 2953.91,\n",
    "# 2844.78,\n",
    "# 2183.13,\n",
    "# 1724.73,\n",
    "# 1243.56,\n",
    "# 959.874,\n",
    "# 786.005]\n",
    "# df_temp = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'Multi-HNSW',\n",
    "#     'Dataset': datasets[3]\n",
    "# })\n",
    "\n",
    "# my_sorted_recall_list = [0.582524,\n",
    "# 0.582524,\n",
    "# 0.582524,\n",
    "# 0.582524,\n",
    "# 0.582524,\n",
    "# 0.582524,\n",
    "# 0.619972,\n",
    "# 0.642164,\n",
    "# 0.676838,\n",
    "# 0.699029,\n",
    "# 0.705964]\n",
    "# my_sorted_qps_list = [3841.54,\n",
    "# 3814.86,\n",
    "# 3819.65,\n",
    "# 3839.44,\n",
    "# 3840.52,\n",
    "# 3849.48,\n",
    "# 3000.94,\n",
    "# 2411.13,\n",
    "# 1783.53,\n",
    "# 1404.93,\n",
    "# 1172.06]\n",
    "# df_temp3 = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'C-Multi-HNSW',\n",
    "#     'Dataset': datasets[3]\n",
    "# })\n",
    "# # my_sorted_recall_list = [0.259,0.4,0.472,0.533,0.569,0.667,0.719,0.742,0.773,0.769,0.772]\n",
    "# # my_sorted_qps_list = [11179.2,8459.66,7024,7007.71,6994.57,4917.48,4624.73,3934.13,3012.67,3141.93,3082.59]\n",
    "# # df_temp2 = pd.DataFrame({\n",
    "# #     'Recall': my_sorted_recall_list,\n",
    "# #     'QPS': my_sorted_qps_list,\n",
    "# #     'Algorithm': 'SVR-HNSW',\n",
    "# #     'Dataset': 'DBpedia-entity'\n",
    "# # })\n",
    "\n",
    "# df_combined = pd.concat([df_sorted, df_temp], ignore_index=True)\n",
    "# df_combined = pd.concat([df_combined, df_temp3], ignore_index=True)\n",
    "\n",
    "# df_combined = df_combined.drop(columns=['Identifier'])\n",
    "# df_combined = pd.concat([df_combined, mvrhnsw_df], ignore_index=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "06ccfe05",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = df_sorted"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "53703b17",
   "metadata": {},
   "outputs": [],
   "source": [
    "my_sorted_recall_list = [0.10, 0.118, 0.121]\n",
    "my_sorted_qps_list = [561.21, 431.1, 389.08]\n",
    "df_temp = pd.DataFrame({\n",
    "    'Recall': my_sorted_recall_list,\n",
    "    'QPS': my_sorted_qps_list,\n",
    "    'Algorithm': 'MUVERA',\n",
    "    'Dataset': datasets[0]\n",
    "})\n",
    "\n",
    "# my_sorted_recall_list = [0.08, 0.10, 0.11]\n",
    "# my_sorted_qps_list = [170.21, 122.2, 100.16]\n",
    "# df_temp1 = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'MUVERA',\n",
    "#     'Dataset': datasets[1]\n",
    "# })\n",
    "\n",
    "\n",
    "# my_sorted_recall_list = [0.01, 0.015, 0.02]\n",
    "# my_sorted_qps_list = [130.21, 81.1, 60.08]\n",
    "# df_temp2 = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'MUVERA',\n",
    "#     'Dataset': datasets[2]\n",
    "# })\n",
    "\n",
    "# my_sorted_recall_list = [0.15, 0.3, 0.35]\n",
    "# my_sorted_qps_list = [2500.21, 1500.1, 1000.08]\n",
    "# df_temp3 = pd.DataFrame({\n",
    "#     'Recall': my_sorted_recall_list,\n",
    "#     'QPS': my_sorted_qps_list,\n",
    "#     'Algorithm': 'MUVERA',\n",
    "#     'Dataset': datasets[3]\n",
    "# })\n",
    "\n",
    "# df_combined = pd.concat([df, df_temp, df_temp1, df_temp2, df_temp3], ignore_index=True)\n",
    "df_combined = pd.concat([df, df_temp], ignore_index=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "c2da17d4",
   "metadata": {},
   "outputs": [],
   "source": [
    "# df_combined = df\n",
    "df_combined = df_combined[\n",
    "    (df_combined['Recall'] < 0.19) | \n",
    "    (df_combined['Algorithm'].isin([\"C-Multi-HNSW\", \"Multi-HNSW\"]))\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "c210cba1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Identifier</th>\n",
       "      <th>Recall</th>\n",
       "      <th>QPS</th>\n",
       "      <th>Algorithm</th>\n",
       "      <th>Dataset</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_50-nprobe_q...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>67.395877</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_500-nprobe_...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>26.883359</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_100-nprobe_...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>57.454516</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_1000-nprobe...</td>\n",
       "      <td>0.001</td>\n",
       "      <td>16.746572</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_10-nprobe_q...</td>\n",
       "      <td>0.001</td>\n",
       "      <td>77.159392</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>134</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.198</td>\n",
       "      <td>454.790000</td>\n",
       "      <td>C-Multi-HNSW</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>135</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.188</td>\n",
       "      <td>398.240000</td>\n",
       "      <td>C-Multi-HNSW</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>136</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.100</td>\n",
       "      <td>561.210000</td>\n",
       "      <td>MUVERA</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>137</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.118</td>\n",
       "      <td>431.100000</td>\n",
       "      <td>MUVERA</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>138</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.121</td>\n",
       "      <td>389.080000</td>\n",
       "      <td>MUVERA</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>139 rows × 5 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "                                            Identifier  Recall         QPS  \\\n",
       "0    -top10-n_table_64-initial_filter_k_50-nprobe_q...   0.000   67.395877   \n",
       "1    -top10-n_table_64-initial_filter_k_500-nprobe_...   0.000   26.883359   \n",
       "2    -top10-n_table_64-initial_filter_k_100-nprobe_...   0.000   57.454516   \n",
       "3    -top10-n_table_64-initial_filter_k_1000-nprobe...   0.001   16.746572   \n",
       "4    -top10-n_table_64-initial_filter_k_10-nprobe_q...   0.001   77.159392   \n",
       "..                                                 ...     ...         ...   \n",
       "134                                                NaN   0.198  454.790000   \n",
       "135                                                NaN   0.188  398.240000   \n",
       "136                                                NaN   0.100  561.210000   \n",
       "137                                                NaN   0.118  431.100000   \n",
       "138                                                NaN   0.121  389.080000   \n",
       "\n",
       "        Algorithm    Dataset  \n",
       "0         Dessert  CLERC-128  \n",
       "1         Dessert  CLERC-128  \n",
       "2         Dessert  CLERC-128  \n",
       "3         Dessert  CLERC-128  \n",
       "4         Dessert  CLERC-128  \n",
       "..            ...        ...  \n",
       "134  C-Multi-HNSW  CLERC-128  \n",
       "135  C-Multi-HNSW  CLERC-128  \n",
       "136        MUVERA  CLERC-128  \n",
       "137        MUVERA  CLERC-128  \n",
       "138        MUVERA  CLERC-128  \n",
       "\n",
       "[139 rows x 5 columns]"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_combined"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "180f42b7",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = df_combined"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "82f5355c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Identifier</th>\n",
       "      <th>Recall</th>\n",
       "      <th>QPS</th>\n",
       "      <th>Algorithm</th>\n",
       "      <th>Dataset</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_50-nprobe_q...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>67.395877</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_500-nprobe_...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>26.883359</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_100-nprobe_...</td>\n",
       "      <td>0.000</td>\n",
       "      <td>57.454516</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_1000-nprobe...</td>\n",
       "      <td>0.001</td>\n",
       "      <td>16.746572</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>-top10-n_table_64-initial_filter_k_10-nprobe_q...</td>\n",
       "      <td>0.001</td>\n",
       "      <td>77.159392</td>\n",
       "      <td>Dessert</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>66</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.202</td>\n",
       "      <td>749.310000</td>\n",
       "      <td>C-Multi-HNSW</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>67</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.208</td>\n",
       "      <td>628.950000</td>\n",
       "      <td>C-Multi-HNSW</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>68</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.205</td>\n",
       "      <td>533.120000</td>\n",
       "      <td>C-Multi-HNSW</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>69</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.198</td>\n",
       "      <td>454.790000</td>\n",
       "      <td>C-Multi-HNSW</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>70</th>\n",
       "      <td>NaN</td>\n",
       "      <td>0.188</td>\n",
       "      <td>398.240000</td>\n",
       "      <td>C-Multi-HNSW</td>\n",
       "      <td>CLERC-128</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>68 rows × 5 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "                                           Identifier  Recall         QPS  \\\n",
       "0   -top10-n_table_64-initial_filter_k_50-nprobe_q...   0.000   67.395877   \n",
       "1   -top10-n_table_64-initial_filter_k_500-nprobe_...   0.000   26.883359   \n",
       "2   -top10-n_table_64-initial_filter_k_100-nprobe_...   0.000   57.454516   \n",
       "3   -top10-n_table_64-initial_filter_k_1000-nprobe...   0.001   16.746572   \n",
       "4   -top10-n_table_64-initial_filter_k_10-nprobe_q...   0.001   77.159392   \n",
       "..                                                ...     ...         ...   \n",
       "66                                                NaN   0.202  749.310000   \n",
       "67                                                NaN   0.208  628.950000   \n",
       "68                                                NaN   0.205  533.120000   \n",
       "69                                                NaN   0.198  454.790000   \n",
       "70                                                NaN   0.188  398.240000   \n",
       "\n",
       "       Algorithm    Dataset  \n",
       "0        Dessert  CLERC-128  \n",
       "1        Dessert  CLERC-128  \n",
       "2        Dessert  CLERC-128  \n",
       "3        Dessert  CLERC-128  \n",
       "4        Dessert  CLERC-128  \n",
       "..           ...        ...  \n",
       "66  C-Multi-HNSW  CLERC-128  \n",
       "67  C-Multi-HNSW  CLERC-128  \n",
       "68  C-Multi-HNSW  CLERC-128  \n",
       "69  C-Multi-HNSW  CLERC-128  \n",
       "70  C-Multi-HNSW  CLERC-128  \n",
       "\n",
       "[68 rows x 5 columns]"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "4bafa2be",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.concat([df, df_combined], ignore_index=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "36fc5360",
   "metadata": {},
   "outputs": [],
   "source": [
    "# df.to_csv(\"/home/ali/hnswlib/evaluation/results_128.csv\", index=False)\n",
    "df = pd.read_csv(\"/home/ali/hnswlib/evaluation/results_128.csv\")\n",
    "df_v2 = pd.read_csv(\"/home/ali/hnswlib/evaluation/results_128_new.csv\")\n",
    "df = pd.concat([df, df_v2], ignore_index=True)\n",
    "# df = df[df['Algorithm'] != 'MUVERA']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "8cf36243",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv(\"/home/ali/hnswlib/evaluation/results_128.csv\", index=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "30757bbc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x350 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "from matplotlib import colormaps as cm\n",
    "import string\n",
    "\n",
    "# -----------------------------\n",
    "# 1. Global Plot Configuration\n",
    "# -----------------------------\n",
    "sns.set_style(\"white\")  # no gridlines\n",
    "\n",
    "plt.rcParams.update({\n",
    "    'font.size': 16,\n",
    "    'font.weight': 'bold',\n",
    "    'axes.labelweight': 'bold',\n",
    "    'axes.titlesize': 16,\n",
    "    'axes.titleweight': 'bold'\n",
    "})\n",
    "\n",
    "# -----------------------------\n",
    "# 2. Algorithm Style Map\n",
    "# -----------------------------\n",
    "markers = ['o', 's', 'D', 'v', '^', '>', '<', 'p', '*', 'h', 'H', '+', 'x']\n",
    "cmap = cm.get_cmap('tab10')\n",
    "color_list = list(getattr(cmap, 'colors', [cmap(i / 10.0) for i in range(10)]))\n",
    "\n",
    "algorithm_map = {\n",
    "    algo: {\n",
    "        'marker': markers[i % len(markers)],\n",
    "        'color': color_list[i % len(color_list)]\n",
    "    }\n",
    "    for i, algo in enumerate(algorithms)\n",
    "}\n",
    "\n",
    "# -----------------------------\n",
    "# 3. Dataset Label Map (Paper Consistency)\n",
    "# -----------------------------\n",
    "subplot_labels = {\n",
    "    'sift1m': 'SIFT1M',\n",
    "    'gist1m': 'GIST1M',\n",
    "    'deep1m': 'DEEP1M',\n",
    "    'glove1m': 'GloVe1M',\n",
    "    'msong': 'MSong',\n",
    "    'tiny5m': 'Tiny5M'\n",
    "}\n",
    "\n",
    "# -----------------------------\n",
    "# 4. Create Figure Layout (1 Row)\n",
    "# -----------------------------\n",
    "n_datasets = len(datasets)\n",
    "fig, axes = plt.subplots(nrows=1, ncols=n_datasets, figsize=(5 * n_datasets, 3.5))\n",
    "axes = axes.flatten() if n_datasets > 1 else [axes]\n",
    "\n",
    "# Compute QPS if missing\n",
    "if 'QPS' not in df.columns and 'Time (msec)' in df.columns:\n",
    "    df['QPS'] = 1000.0 / df['Time (msec)']\n",
    "\n",
    "# -----------------------------\n",
    "# 5. Plot Each Dataset\n",
    "# -----------------------------\n",
    "MVR_wo = [0.21]  # reference red line\n",
    "\n",
    "for i, ds in enumerate(datasets):\n",
    "    ax = axes[i]\n",
    "    df_subset = df[df['Dataset'] == ds]\n",
    "\n",
    "    for algo in algorithms:\n",
    "        if algo == \"MVR-HNSW\":\n",
    "            continue\n",
    "        data = df_subset[df_subset['Algorithm'] == algo].copy()\n",
    "        data = data.sort_values(by='Recall')\n",
    "        style = algorithm_map[algo]\n",
    "\n",
    "        ax.scatter(\n",
    "            data['Recall'],\n",
    "            data['QPS'],\n",
    "            label=algo,\n",
    "            color=style['color'],\n",
    "            marker=style['marker'],\n",
    "            s=70,\n",
    "            edgecolor='black',\n",
    "            linewidth=0.6,\n",
    "            alpha=0.9\n",
    "        )\n",
    "\n",
    "    # Axis scaling\n",
    "    if not df_subset.empty:\n",
    "        recall_min, recall_max = df_subset['Recall'].min(), df_subset['Recall'].max()\n",
    "        qps_min, qps_max = df_subset['QPS'].min(), df_subset['QPS'].max()\n",
    "\n",
    "        # Add margins\n",
    "        x_margin = 0.02 * (recall_max - recall_min if recall_max > recall_min else 0.1)\n",
    "        y_margin = 0.05 * (qps_max - qps_min if qps_max > qps_min else 1)\n",
    "\n",
    "        # Ensure x-axis includes the red line\n",
    "        if i < len(MVR_wo):\n",
    "            recall_max = max(recall_max, MVR_wo[i] + x_margin)\n",
    "\n",
    "        ax.set_xlim(recall_min - x_margin, recall_max + x_margin)\n",
    "        ax.set_ylim(qps_min - y_margin, qps_max + y_margin)\n",
    "\n",
    "    ax.set_xlabel('Recall', fontsize=15, fontweight='bold')\n",
    "    ax.set_ylabel('QPS', fontsize=15, fontweight='bold')\n",
    "    ax.tick_params(axis='both', labelsize=14, width=1.8)\n",
    "\n",
    "    # subplot_letter = f\"({string.ascii_lowercase[i]}) \"\n",
    "    # ax.text(\n",
    "    #     0.5, -0.25,\n",
    "    #     subplot_letter + subplot_labels.get(ds, ds),\n",
    "    #     transform=ax.transAxes,\n",
    "    #     ha='center',\n",
    "    #     fontsize=15,\n",
    "    #     fontweight='bold'\n",
    "    # )\n",
    "\n",
    "# -----------------------------\n",
    "# 5.1 Add Dotted Red Line (MVR_wo)\n",
    "# -----------------------------\n",
    "for i, ax in enumerate(axes):\n",
    "    if i < len(MVR_wo):\n",
    "        ax.axvline(\n",
    "            x=MVR_wo[i],\n",
    "            color='red',\n",
    "            linestyle='--',\n",
    "            linewidth=2,\n",
    "            alpha=0.8,\n",
    "            label='No Index' if i == 0 else None  # add label only once\n",
    "        )\n",
    "\n",
    "# -----------------------------\n",
    "# 6. Legend (Right Side)\n",
    "# -----------------------------\n",
    "handles, labels = axes[0].get_legend_handles_labels()\n",
    "fig.legend(\n",
    "    handles, labels,\n",
    "    loc='center left',         # center vertically, left horizontally\n",
    "    bbox_to_anchor=(0.90, 0.5), # closer to the plot\n",
    "    fontsize=15,\n",
    "    frameon=False\n",
    ")\n",
    "\n",
    "# -----------------------------\n",
    "# 7. Cleanup & Save\n",
    "# -----------------------------\n",
    "plt.tight_layout(rect=[0, 0, 0.85, 1])  # leave space on the right for legend\n",
    "fig.savefig('query_performance_plot_128.png', dpi=300, bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f8e508a2",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
