{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "b2576d17-5103-44d6-91b0-3e81ee5c9779",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "13818.0\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>final_best_mean</th>\n",
       "      <th>final_best_std</th>\n",
       "      <th>auc_best_so_far_mean</th>\n",
       "      <th>auc_best_so_far_std</th>\n",
       "      <th>final_simple_regret_mean</th>\n",
       "      <th>final_simple_regret_std</th>\n",
       "      <th>evals_to_threshold_mean</th>\n",
       "      <th>evals_to_threshold_std</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>qap_botorch_mallows_EI_benchmark_index_3</td>\n",
       "      <td>21908.4</td>\n",
       "      <td>4101.909048</td>\n",
       "      <td>2750160.05</td>\n",
       "      <td>744154.991184</td>\n",
       "      <td>8090.4</td>\n",
       "      <td>4101.909048</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>qap_botorch_merge_EI_benchmark_index_3</td>\n",
       "      <td>28794.6</td>\n",
       "      <td>5557.929438</td>\n",
       "      <td>3833182.85</td>\n",
       "      <td>848461.864423</td>\n",
       "      <td>14976.6</td>\n",
       "      <td>5557.929438</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>qap_botorch_merge_EI_benchmark_index_pairwise_...</td>\n",
       "      <td>22098.6</td>\n",
       "      <td>4536.022138</td>\n",
       "      <td>2975522.95</td>\n",
       "      <td>743481.970657</td>\n",
       "      <td>8280.6</td>\n",
       "      <td>4536.022138</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>qap_botorch_merge_EI_benchmark_index_shift_pai...</td>\n",
       "      <td>20246.2</td>\n",
       "      <td>3777.596532</td>\n",
       "      <td>2580381.55</td>\n",
       "      <td>838626.704865</td>\n",
       "      <td>6428.2</td>\n",
       "      <td>3777.596532</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>qap_botorch_merge_EI_benchmark_index_shift_pai...</td>\n",
       "      <td>20396.0</td>\n",
       "      <td>4579.187308</td>\n",
       "      <td>2493028.75</td>\n",
       "      <td>690320.715409</td>\n",
       "      <td>6578.0</td>\n",
       "      <td>4579.187308</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>qap_botorch_merge_EI_benchmark_index_shift_pat...</td>\n",
       "      <td>20148.0</td>\n",
       "      <td>3132.213275</td>\n",
       "      <td>2588481.35</td>\n",
       "      <td>638766.734605</td>\n",
       "      <td>6330.0</td>\n",
       "      <td>3132.213275</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "                                           algorithm  final_best_mean  \\\n",
       "0           qap_botorch_mallows_EI_benchmark_index_3          21908.4   \n",
       "1             qap_botorch_merge_EI_benchmark_index_3          28794.6   \n",
       "2  qap_botorch_merge_EI_benchmark_index_pairwise_...          22098.6   \n",
       "3  qap_botorch_merge_EI_benchmark_index_shift_pai...          20246.2   \n",
       "4  qap_botorch_merge_EI_benchmark_index_shift_pai...          20396.0   \n",
       "5  qap_botorch_merge_EI_benchmark_index_shift_pat...          20148.0   \n",
       "\n",
       "   final_best_std  auc_best_so_far_mean  auc_best_so_far_std  \\\n",
       "0     4101.909048            2750160.05        744154.991184   \n",
       "1     5557.929438            3833182.85        848461.864423   \n",
       "2     4536.022138            2975522.95        743481.970657   \n",
       "3     3777.596532            2580381.55        838626.704865   \n",
       "4     4579.187308            2493028.75        690320.715409   \n",
       "5     3132.213275            2588481.35        638766.734605   \n",
       "\n",
       "   final_simple_regret_mean  final_simple_regret_std evals_to_threshold_mean  \\\n",
       "0                    8090.4              4101.909048                    None   \n",
       "1                   14976.6              5557.929438                    None   \n",
       "2                    8280.6              4536.022138                    None   \n",
       "3                    6428.2              3777.596532                    None   \n",
       "4                    6578.0              4579.187308                    None   \n",
       "5                    6330.0              3132.213275                    None   \n",
       "\n",
       "  evals_to_threshold_std  \n",
       "0                   None  \n",
       "1                   None  \n",
       "2                   None  \n",
       "3                   None  \n",
       "4                   None  \n",
       "5                   None  "
      ]
     },
     "execution_count": 1,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "from copy import deepcopy\n",
    "\n",
    "def read_file(kernel_type, dim, benchmark_index, nruns):\n",
    "    file_name = 'tsp_botorch_'+kernel_type+'_EI_dim_'+str(dim)+'benchmark_index_'+str(benchmark_index)+'_nrun_'+str(nruns)+'.pkl'\n",
    "    data = torch.load(file_name, weights_only=False)\n",
    "    l = data['outputs']\n",
    "    l = [float(i) for i in l]\n",
    "    return l\n",
    "\n",
    "\n",
    "def read_file_filename(file_name):\n",
    "    data = torch.load(file_name, weights_only=False)\n",
    "    l = data['outputs']\n",
    "    l = [float(i) for i in l]\n",
    "    return l\n",
    "\n",
    "\n",
    "def read_file_no_anchor(kernel_type, dim, benchmark_index, nruns):\n",
    "    file_name = 'tsp_botorch_'+kernel_type+'_EI_dim_'+str(dim)+'benchmark_index_no_anchor_'+str(benchmark_index)+'_nrun_'+str(nruns)+'.pkl'\n",
    "    data = torch.load(file_name, weights_only=False)\n",
    "    l = data['outputs']\n",
    "    l = [float(i) for i in l]\n",
    "    return l\n",
    "\n",
    "\n",
    "import os\n",
    "\n",
    "def analyse_trial(dim=10, benchmark_index=0):\n",
    "    folders = os.listdir('./results')\n",
    "    nruns = 20\n",
    "    results_dict = {}\n",
    "\n",
    "    for folder in folders:\n",
    "        if '.' in folder:\n",
    "            continue\n",
    "        results_dict[folder] = []\n",
    "        for nrun in range(nruns):\n",
    "            results_dict[folder].append(read_file_filename(os.path.join('./results', folder, folder+f'_nrun_{nrun}.pkl')))\n",
    "\n",
    "    all_results = []\n",
    "    for key in results_dict.keys():\n",
    "        results_dict[key] = np.array(results_dict[key])\n",
    "        all_results.append(results_dict[key])\n",
    "    # print(all_results[0].shape)\n",
    "    # return all_results\n",
    "    global_minimum = np.min(all_results)\n",
    "    print(global_minimum)\n",
    "    best_so_far = [np.minimum.accumulate(res, axis=1) for res in all_results]\n",
    "    regrets = [bfs - global_minimum for bfs in best_so_far]\n",
    "    for i, key in enumerate(results_dict.keys()):\n",
    "        # results_dict[key] = regrets[i]\n",
    "        results_dict[key] = all_results[i]\n",
    "    return results_dict\n",
    "\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from sklearn.metrics import auc\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from sklearn.metrics import auc\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from sklearn.metrics import auc\n",
    "\n",
    "def evaluate_algorithms(r: dict, f_opt=None, threshold=None):\n",
    "    \"\"\"\n",
    "    r: dict of {algorithm_name: np.ndarray of shape (n_repeats, n_iterations)}\n",
    "    f_opt: known global minimum value (float)\n",
    "    threshold: optional regret threshold to measure how many iterations are needed\n",
    "    \n",
    "    Returns:\n",
    "        pd.DataFrame with aggregated metrics for each algorithm\n",
    "    \"\"\"\n",
    "    results = []\n",
    "\n",
    "    for algo, outputs in r.items():\n",
    "        outputs = np.array(outputs)  # shape: (n_repeats, n_iterations)\n",
    "        n_repeats, n_iterations = outputs.shape\n",
    "\n",
    "        best_so_far = np.minimum.accumulate(outputs, axis=1)\n",
    "        \n",
    "        final_best = best_so_far[:, -1]\n",
    "        auc_vals = np.array([\n",
    "            auc(np.arange(1, n_iterations+1), best_so_far[i] - f_opt)\n",
    "            for i in range(n_repeats)\n",
    "        ])\n",
    "\n",
    "        metrics = {\n",
    "            \"algorithm\": algo,\n",
    "            \"final_best_mean\": np.mean(final_best),\n",
    "            \"final_best_std\": np.std(final_best),\n",
    "            \"auc_best_so_far_mean\": np.mean(auc_vals),\n",
    "            \"auc_best_so_far_std\": np.std(auc_vals),\n",
    "        }\n",
    "\n",
    "        if f_opt is not None:\n",
    "            simple_regrets       = best_so_far - f_opt       # ← 修正\n",
    "            instantaneous_regret = outputs - f_opt           # (可选) 如需两者都存\n",
    "            final_simple         = simple_regrets[:, -1]\n",
    "\n",
    "            metrics.update({\n",
    "                \"final_simple_regret_mean\": np.mean(final_simple),\n",
    "                \"final_simple_regret_std\": np.std(final_simple),\n",
    "            })\n",
    "        else:\n",
    "            metrics.update({\n",
    "                \"final_simple_regret_mean\": None,\n",
    "                \"final_simple_regret_std\": None,\n",
    "            })\n",
    "\n",
    "        if threshold is not None:\n",
    "            evals_to_threshold = []\n",
    "            for i in range(n_repeats):\n",
    "                for j in range(n_iterations):\n",
    "                    if best_so_far[i, j] <= threshold:\n",
    "                        evals_to_threshold.append(j + 1)\n",
    "                        break\n",
    "                else:\n",
    "                    evals_to_threshold.append(n_iterations)\n",
    "            evals_to_threshold = np.array(evals_to_threshold)\n",
    "            metrics.update({\n",
    "                \"evals_to_threshold_mean\": np.mean(evals_to_threshold),\n",
    "                \"evals_to_threshold_std\": np.std(evals_to_threshold),\n",
    "            })\n",
    "        else:\n",
    "            metrics.update({\n",
    "                \"evals_to_threshold_mean\": None,\n",
    "                \"evals_to_threshold_std\": None,\n",
    "            })\n",
    "\n",
    "        results.append(metrics)\n",
    "\n",
    "    return pd.DataFrame(results)\n",
    "\n",
    "\n",
    "r = analyse_trial()\n",
    "k = evaluate_algorithms(r, 13818.0)\n",
    "k"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1ecd93d1-d9c5-4faa-a690-834f938cb6d0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "13818.0\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# ---------- 1. 读取并整理数据 ----------\n",
    "r    = analyse_trial()   # 你的函数\n",
    "results = {}\n",
    "results['Mallows'] = r['qap_botorch_mallows_EI_benchmark_index_3']\n",
    "results['Merge'] = r['qap_botorch_merge_EI_benchmark_index_3']\n",
    "\n",
    "\n",
    "global_min = np.inf\n",
    "for arr in results.values():\n",
    "    global_min = min(global_min, arr[:, :].min())\n",
    "\n",
    "# 把每个算法的 simple regret（best‑so‑far – global_min）整理成均值±std\n",
    "mean_regret = {}\n",
    "std_regret  = {}\n",
    "\n",
    "for algo, outputs in results.items():\n",
    "    outputs        = outputs[:, :]                          \n",
    "    best_so_far    = np.minimum.accumulate(outputs, axis=1)    \n",
    "    regrets        = best_so_far - global_min                 \n",
    "    mean_regret[algo] = regrets.mean(axis=0)             \n",
    "    std_regret[algo]  = regrets.std(axis=0)\n",
    "\n",
    "# ---------- 2. 画图 ----------\n",
    "iters = np.arange(1, 201)\n",
    "\n",
    "plt.figure(figsize=(5, 4))\n",
    "\n",
    "for algo in mean_regret:\n",
    "    mean = mean_regret[algo]\n",
    "    std  = std_regret[algo]\n",
    "    plt.plot(iters, mean, label=algo)\n",
    "    plt.fill_between(iters, mean - std, mean + std, alpha=0.2)\n",
    "\n",
    "plt.xlabel(\"Iteration\")\n",
    "plt.ylabel(\"Simple Regret (mean ± 1 std)\")\n",
    "# plt.yscale(\"log\")           # 若差距跨数量级，建议用对数轴；可去掉\n",
    "plt.legend()\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"qap_curve.pdf\", dpi=600, bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "074cef91-4724-417b-8c7a-a34fa89836cb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "16566.0\n",
      "dict_keys(['qap_botorch_mallows_EI_benchmark_index_3', 'qap_botorch_merge_EI_benchmark_index_3'])\n",
      "win: 1, tie: 3, loss: 16\n",
      "16566.0\n",
      "win: 1, tie: 2, loss: 17\n"
     ]
    }
   ],
   "source": [
    "\n",
    "def analyse_trial(dim=10, benchmark_index=0):\n",
    "    folders = os.listdir('./results')[:2]\n",
    "    nruns = 20\n",
    "    results_dict = {}\n",
    "\n",
    "    for folder in folders:\n",
    "        if '.' in folder:\n",
    "            continue\n",
    "        results_dict[folder] = []\n",
    "        for nrun in range(nruns):\n",
    "            results_dict[folder].append(read_file_filename(os.path.join('./results', folder, folder+f'_nrun_{nrun}.pkl')))\n",
    "\n",
    "    all_results = []\n",
    "    for key in results_dict.keys():\n",
    "        results_dict[key] = np.array(results_dict[key])\n",
    "        all_results.append(results_dict[key])\n",
    "    # print(all_results[0].shape)\n",
    "    # return all_results\n",
    "    global_minimum = np.min(all_results)\n",
    "    print(global_minimum)\n",
    "    best_so_far = [np.minimum.accumulate(res, axis=1) for res in all_results]\n",
    "    regrets = [bfs - global_minimum for bfs in best_so_far]\n",
    "    for i, key in enumerate(results_dict.keys()):\n",
    "        results_dict[key] = regrets[i]\n",
    "        # results_dict[key] = all_results[i]\n",
    "    return results_dict\n",
    "\n",
    "# analyse_trial()\n",
    "\n",
    "def final_regret_count(d):\n",
    "    print(d.keys())\n",
    "    mallows = d[list(d.keys())[0]]\n",
    "    merge = d[list(d.keys())[1]]\n",
    "    win = 0\n",
    "    tie = 0\n",
    "    loss = 0\n",
    "    for i in range(20):\n",
    "        if mallows[i, -1] > merge[i, -1]:\n",
    "            win += 1\n",
    "        elif mallows[i, -1] == merge[i, -1]:\n",
    "            tie += 1\n",
    "        else:\n",
    "            loss += 1\n",
    "    print(f'win: {win}, tie: {tie}, loss: {loss}')\n",
    "\n",
    "\n",
    "def cumulative_regret_count(d):\n",
    "    mallows = d[list(d.keys())[0]]\n",
    "    merge = d[list(d.keys())[1]]\n",
    "    win = 0\n",
    "    tie = 0\n",
    "    loss = 0\n",
    "    for i in range(20):\n",
    "        if mallows[i].mean() > merge[i].mean():\n",
    "            win += 1\n",
    "        elif mallows[i].mean() == merge[i].mean():\n",
    "            tie += 1\n",
    "        else:\n",
    "            loss += 1\n",
    "    print(f'win: {win}, tie: {tie}, loss: {loss}')\n",
    "\n",
    "final_regret_count(analyse_trial())\n",
    "cumulative_regret_count(analyse_trial())"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
