{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "bf27d10f-649f-4c0a-b787-0159e05fd95e",
   "metadata": {},
   "source": [
    "## Analyse results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "02fe50e1-f3ce-46ee-948c-0d8cea009fcc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "329.969018\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>tsp_botorch_mallows_EI_dim_10benchmark_index_0</td>\n",
       "      <td>329.981921</td>\n",
       "      <td>0.038710</td>\n",
       "      <td>428.206103</td>\n",
       "      <td>121.906771</td>\n",
       "      <td>0.012903</td>\n",
       "      <td>0.038710</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>tsp_botorch_merge_EI_dim_10benchmark_index_0</td>\n",
       "      <td>330.046437</td>\n",
       "      <td>0.125101</td>\n",
       "      <td>527.600875</td>\n",
       "      <td>162.856862</td>\n",
       "      <td>0.077419</td>\n",
       "      <td>0.125101</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>tsp_botorch_merge_EI_dim_10benchmark_index_pai...</td>\n",
       "      <td>329.988373</td>\n",
       "      <td>0.046074</td>\n",
       "      <td>410.827703</td>\n",
       "      <td>107.345022</td>\n",
       "      <td>0.019355</td>\n",
       "      <td>0.046074</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>tsp_botorch_merge_EI_dim_10benchmark_index_shi...</td>\n",
       "      <td>330.007728</td>\n",
       "      <td>0.059130</td>\n",
       "      <td>471.653413</td>\n",
       "      <td>167.540065</td>\n",
       "      <td>0.038710</td>\n",
       "      <td>0.059130</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>tsp_botorch_merge_EI_dim_10benchmark_index_shi...</td>\n",
       "      <td>329.981921</td>\n",
       "      <td>0.038710</td>\n",
       "      <td>400.905878</td>\n",
       "      <td>115.012314</td>\n",
       "      <td>0.012903</td>\n",
       "      <td>0.038710</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>tsp_botorch_merge_EI_dim_10benchmark_index_shi...</td>\n",
       "      <td>329.994824</td>\n",
       "      <td>0.051613</td>\n",
       "      <td>418.782449</td>\n",
       "      <td>125.345903</td>\n",
       "      <td>0.025806</td>\n",
       "      <td>0.051613</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     tsp_botorch_mallows_EI_dim_10benchmark_index_0       329.981921   \n",
       "1       tsp_botorch_merge_EI_dim_10benchmark_index_0       330.046437   \n",
       "2  tsp_botorch_merge_EI_dim_10benchmark_index_pai...       329.988373   \n",
       "3  tsp_botorch_merge_EI_dim_10benchmark_index_shi...       330.007728   \n",
       "4  tsp_botorch_merge_EI_dim_10benchmark_index_shi...       329.981921   \n",
       "5  tsp_botorch_merge_EI_dim_10benchmark_index_shi...       329.994824   \n",
       "\n",
       "   final_best_std  auc_best_so_far_mean  auc_best_so_far_std  \\\n",
       "0        0.038710            428.206103           121.906771   \n",
       "1        0.125101            527.600875           162.856862   \n",
       "2        0.046074            410.827703           107.345022   \n",
       "3        0.059130            471.653413           167.540065   \n",
       "4        0.038710            400.905878           115.012314   \n",
       "5        0.051613            418.782449           125.345903   \n",
       "\n",
       "   final_simple_regret_mean  final_simple_regret_std evals_to_threshold_mean  \\\n",
       "0                  0.012903                 0.038710                    None   \n",
       "1                  0.077419                 0.125101                    None   \n",
       "2                  0.019355                 0.046074                    None   \n",
       "3                  0.038710                 0.059130                    None   \n",
       "4                  0.012903                 0.038710                    None   \n",
       "5                  0.025806                 0.051613                    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, 329.969018)\n",
    "k"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "358182cf-3420-4b40-9646-a459a72f770f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "329.969018\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",
    "cut        = 200     # 要展示的最大迭代步数\n",
    "trial_cut  = 20      # 每个算法取前 trial_cut 条独立运行\n",
    "r    = analyse_trial(cut=cut, trial_cut=trial_cut)   # 你的函数\n",
    "results = {}\n",
    "results['Mallows'] = r['tsp_botorch_mallows_EI_dim_10benchmark_index_0']\n",
    "results['Merge'] = r['tsp_botorch_merge_EI_dim_10benchmark_index_shift_pairwise_pattern_0']\n",
    "\n",
    "# 计算全局最优值（所有算法、所有运行、到 cut 步为止的最小值）\n",
    "global_min = np.inf\n",
    "for arr in results.values():\n",
    "    global_min = min(global_min, arr[:, :cut].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[:, :cut]                          # shape = (n_runs, cut)\n",
    "    best_so_far    = np.minimum.accumulate(outputs, axis=1)    # shape = (n_runs, cut)\n",
    "    regrets        = best_so_far - global_min                  # simple regret\n",
    "    mean_regret[algo] = regrets.mean(axis=0)                   # (cut,)\n",
    "    std_regret[algo]  = regrets.std(axis=0)\n",
    "\n",
    "# ---------- 2. 画图 ----------\n",
    "iters = np.arange(1, cut + 1)\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(\"cp_curve.pdf\", dpi=600, bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  }
 ],
 "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
}
