{
 "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": 5,
   "id": "358182cf-3420-4b40-9646-a459a72f770f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "329.969018\n"
     ]
    },
    {
     "data": {
      "image/png": "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6enpABQWFtKjRw//9sLCQs4888xGz7PZbNhs4U+YDV7bHh30utTB6tH3DPZ93Yu+fE/R16voPebKZs+pdHkod3iIszfzcR/bD8n9wRodmmCFEEK0WYsS9VVXXUVBQQHdu3cPuH3qRKHso+7bty/p6emsWrXKn5jLysrYsGEDt956a0iuEWqarmONisFZFZpbtBq7xt7ul9C3cBlnH3qN/f9ex6G0i+g7+rImzztS4STaaqapO7ZQBhzd60vW0mcthBARoUWJ2qgz97TRhnmoT1RRUcGePXv8z/ft28eWLVtISkoiKyuLO+64gwcffJD+/fvTt29f7r33XjIyMpr8stDR7DEJYU3UACmDL+FowVskaWUM9u5g8KEd/Gudm4HnXtHoBDQeQ3G00klqXDOtDYbHl6yT+oElPH3uQgghWi7o27NCac2aNVx00UX1tt90000sX77cP+HJiy++SElJCeeddx7PPfccAwYMaPE12vP2LICykmIK928P+3Wc1VUcLdiPvm8t55X9B0NprOw5j2GjL2z0HA3ITIrGZm7h7fNRSZCQecJyXEIIIdoqmNzU4kSdk5NDcXExl19+uX/bX//6VxYuXEhlZSVXXXUVzzzzTIf1DzemvRO11+Phu69zwn4dP6UoXfssI0s+oBobhRctxp3Q+IA/q1mnV2J0003gdcV0l6lGhRAixMIyM9kDDzzAN99843++bds2Zs6cyfjx47n77rv597//zaJFi1ofdRdhMpux2tpxMJamYTv3Vj41hhCFk+5fPIrmaXzmNpfHoLAsiJndKg+DW5Y1FUKIjtLiRL1lyxbGjRvnf75ixQrGjBnD0qVLmTt3Lk8//TRvvPFGWILsbGwx4a+512W3mPlT1CyKVCLRlXmkfLOsyeMrXR7ySx1NTzHqp6D0+9AEKoQQImgtTtTHjh0LmHxk7dq1TJo0yf981KhRIZsDvLOLaudEDZCalMwd7l8BkLDvPSzlTX8WFU4PB45UcqzK3XzhrgrwOJs/TgghRMi1OFGnpaX5ZyVzuVxs2rSJs88+27+/vLy83rzfJ6uY+G7tvvzngCQTnxmD+cIyCg2DlO2vNHuOwnfb1qESB57matfOitAEKoQQIigtTtSXXXYZd999N59++ikLFiwgOjqa888/379/69atnHLKKWEJsrMxW6zYouPa9Zr9k3wLazzsug6F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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['tsp_botorch_mallows_EI_dim_10benchmark_index_0']\n",
    "results['Merge'] = r['tsp_botorch_merge_EI_dim_10benchmark_index_0']\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(\"tsp_curve.pdf\", dpi=600, bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b919d908-f18f-449f-8a2d-cde85d48bbb1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "329.969018\n",
      "dict_keys(['tsp_botorch_mallows_EI_dim_10benchmark_index_0', 'tsp_botorch_merge_EI_dim_10benchmark_index_0'])\n",
      "win: 1, tie: 12, loss: 7\n",
      "329.969018\n",
      "win: 5, tie: 0, loss: 15\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
}
