{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "f9488384-3238-49cb-ae5d-0d5011219bda",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "62.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>pl_botorch_mallows_EI_30</td>\n",
       "      <td>68.05</td>\n",
       "      <td>3.040970</td>\n",
       "      <td>5350.500</td>\n",
       "      <td>910.120116</td>\n",
       "      <td>6.05</td>\n",
       "      <td>3.040970</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>pl_botorch_merge_EI_30</td>\n",
       "      <td>68.05</td>\n",
       "      <td>2.268810</td>\n",
       "      <td>5495.650</td>\n",
       "      <td>687.715168</td>\n",
       "      <td>6.05</td>\n",
       "      <td>2.268810</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>pl_botorch_merge_EI_30_pairwise_pattern</td>\n",
       "      <td>68.35</td>\n",
       "      <td>3.691544</td>\n",
       "      <td>5590.600</td>\n",
       "      <td>1040.549081</td>\n",
       "      <td>6.35</td>\n",
       "      <td>3.691544</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>pl_botorch_merge_EI_30_shift_pairwise</td>\n",
       "      <td>68.05</td>\n",
       "      <td>2.418160</td>\n",
       "      <td>5390.300</td>\n",
       "      <td>952.258452</td>\n",
       "      <td>6.05</td>\n",
       "      <td>2.418160</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>pl_botorch_merge_EI_30_shift_pairwise_pattern</td>\n",
       "      <td>68.10</td>\n",
       "      <td>2.142429</td>\n",
       "      <td>5371.025</td>\n",
       "      <td>792.776064</td>\n",
       "      <td>6.10</td>\n",
       "      <td>2.142429</td>\n",
       "      <td>None</td>\n",
       "      <td>None</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>pl_botorch_merge_EI_30_shift_pattern</td>\n",
       "      <td>67.15</td>\n",
       "      <td>2.127792</td>\n",
       "      <td>5553.150</td>\n",
       "      <td>1096.469004</td>\n",
       "      <td>5.15</td>\n",
       "      <td>2.127792</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                       pl_botorch_mallows_EI_30            68.05   \n",
       "1                         pl_botorch_merge_EI_30            68.05   \n",
       "2        pl_botorch_merge_EI_30_pairwise_pattern            68.35   \n",
       "3          pl_botorch_merge_EI_30_shift_pairwise            68.05   \n",
       "4  pl_botorch_merge_EI_30_shift_pairwise_pattern            68.10   \n",
       "5           pl_botorch_merge_EI_30_shift_pattern            67.15   \n",
       "\n",
       "   final_best_std  auc_best_so_far_mean  auc_best_so_far_std  \\\n",
       "0        3.040970              5350.500           910.120116   \n",
       "1        2.268810              5495.650           687.715168   \n",
       "2        3.691544              5590.600          1040.549081   \n",
       "3        2.418160              5390.300           952.258452   \n",
       "4        2.142429              5371.025           792.776064   \n",
       "5        2.127792              5553.150          1096.469004   \n",
       "\n",
       "   final_simple_regret_mean  final_simple_regret_std evals_to_threshold_mean  \\\n",
       "0                      6.05                 3.040970                    None   \n",
       "1                      6.05                 2.268810                    None   \n",
       "2                      6.35                 3.691544                    None   \n",
       "3                      6.05                 2.418160                    None   \n",
       "4                      6.10                 2.142429                    None   \n",
       "5                      5.15                 2.127792                    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": 3,
     "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, 62)\n",
    "k"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bf27d10f-649f-4c0a-b787-0159e05fd95e",
   "metadata": {},
   "source": [
    "## Analyse results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "2f7f19a5-b3c2-497b-8b34-33ee8539b596",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "62.0\n"
     ]
    },
    {
     "data": {
      "image/png": "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TH76058yfvx4nnnmmYir4dtKStRRFpAStRCiA9hMBrY8fnGHvXZb3HLLLUyfPh2g0eWOL730Unr16sVbb71Fbm4umqYxYMCAqK4WNnbsWN58802WLVvGkCFDSEpKCiXvJUuWMGbMmNCxjz76KNdddx3/+c9/+PTTT3nkkUd4//33ueKKK6IWT2MkUUeZLm3UQogOoChKxNXPHe2SSy7B6/WiKAoXXxx+k1FeXs62bdt46623OO+88wBYvnx5RNc//fTTWbFiRdi2FStW0Ldv39Dc22PGjOGee+7h73//e6gteuzYsXzxxResWLGCe++9N+z8vn370rdvX2bMmMHkyZOZO3euJOquRpabE0KI1jEYDGzdujX089FSU1NJT0/nzTffJCcnh6KiIh544IGIrn/vvfcyYsQInnjiCX7yk59QWFjIq6++yuuvvx46ZtCgQaSmpvLee+8xf/58IJioZ82ahaIooarz2tpafvnLX3L11VdTUFDA/v37WbVqVYN29FiQNuook17fQgjReklJSSQlJTXYrqoq77//PmvWrGHAgAHMmDGD559/PqJrDx06lA8++ID333+fAQMG8PDDD/P444+HTaiiKArnnXceiqJw7rnnAsHknZSUxPDhw4mPjweCNxLl5eXceOON9O3bl2uuuYYJEyYcl5UYFb0t/epPMNXV1SQnJ2O32xv9g4nEoeI9ZObmRycwIYRogtvtZvfu3RQUFDRY/ld0Ds39jiLJO1KijgEZoiWEECJaJFHHgFR/CyGEiBZJ1DEgiVoIIUS0SKKOAV16fgshhIgSSdQxICVqIYQQ0SKJOgZkdjIhhBDRIok6FmS+byGEEFEiiToGpOpbCCFEtEiijgFZQUsIIUS0SKKOAen1LYQQjZs6dSqKonDHHXc02Ddt2jQURQmb4lNIoo4JqfoWQoim5eXl8f7771NbWxva5na7ee+99+jZs2ebr6vrOn7/iff9K4k6BmQKUSGEaNrQoUPJy8vjww8/DG378MMP6dmzJ0OGDAlt0zSNOXPmUFBQgM1mY/DgwfzjH/8I7V+8eDGKovDpp58ybNgwLBYLy5cvp6amhilTphAfH09OTg6//e1vGTt2LPfcc0/oXI/Hw6xZs+jevTvx8fGMGjWKxYsXH4+3HzFJ1DEgVd9CCNG8W265hblz54aev/3229x8881hx8yZM4c/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GbVY2qK5aUSbEz9yCpV6Aqfoezm44t0oRxVUP5YaoLrWj73WF5PXEUII0X4RJ+p//vOfXHzxxdhsNtatW4fHExyXa7fbefrpp6MWWGlpKQDdunUL296tW7fQvtLSUrKyssL2G41G0tLSQsc0xuPxUF1dHfaINrWJpS5bkpiczto+dwJwXuW/KN37XTTDAo60UdcrsddG/TWEEEJER8SJ+sknn+SNN97grbfeCls565xzzukys5LNmTOH5OTk0CMvLy/qr6FGMJb6WD0HjWGl5WyMikbixj9FMaog5ZhE7fPruLzSsUwIITqjiBP1tm3bOP/88xtsT05OpqqqKhoxAcFhYAAHDx4M237w4MHQvuzsbMrKysL2+/1+KioqQsc0Zvbs2djt9tBj3759UYu7ntnavuUkleE349MNnOnfQHG0O5bpfjhmbnGHWxK1EEJ0Rm0aR71jx44G25cvX07v3r2jEhRAQUEB2dnZLFy4MLSturqalStXMnr0aABGjx5NVVUVa9asCR3z5Zdfomkao0aNavLaFouFpKSksEe0WWwJ7To/NTOHlfEXBH/e+l7UF+1Q/eEd3aolUQshRKcUcaK+7bbb+MUvfsHKlStRFIXi4mLeffddZs2axZ133hnRtRwOB+vXr2f9+vVAsAPZ+vXrKSoqQlEU7rnnHp588kk++eQTNm7cyI033khubm6oZ/jpp5/OJZdcwm233cY333zDihUrmD59Otdee22re3zHii2ufYkawDJ0Ml7dwKDAZvp8fAV8PJ0ae0UUogPV5wx7XusN4A/IUC0hhOhsIu7x9MADD6BpGuPGjcPlcnH++edjsViYNWsWd999d0TXWr16NRdccEHo+cyZMwG46aabmDdvHvfddx9Op5Pbb7+dqqoqzj33XD777DOsVmvonHfffZfp06czbtw4VFXlqquu4ne/+12kbyvqjCYzBqOZgN/b8sFNSMnoxvK0Kxlb8Q+Misap+h6WrP6AxHF3tDs+g7cmNESrnsPjJyUu+sPBhBBCtF2bx1F7vV527NiBw+Ggf//+JCS0vwTZUWIxjhpg/45N1Doq230dv99H8davGbfzWdy6ic0XvEVSSka7r+tO6w9H9U5PiTORlxbX7usKIYRoXszHUQOYzWb69+/PyJEju3SSjiVLFKq/AYxGE3lnnMMWtS9WxYdj9QdRua7qC59q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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['pl_botorch_mallows_EI_30']\n",
    "results['Merge'] = r['pl_botorch_merge_EI_30']\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(\"cp_curve.pdf\", dpi=600, bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "68039be9-c6c5-4275-9178-4027709f3d7f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "64.0\n",
      "dict_keys(['pl_botorch_mallows_EI_30', 'pl_botorch_merge_EI_30'])\n",
      "win: 9, tie: 3, loss: 8\n",
      "64.0\n",
      "win: 8, tie: 0, loss: 12\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())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "25c7e88a-62f9-4f37-9dc2-6b2f1468a30a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([1, 2009])"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "from copy import deepcopy\n",
    "\n",
    "\n",
    "def merge_sort(arr):\n",
    "    # print(f'Handling array: {arr}')\n",
    "    if len(arr) == 1:\n",
    "        return []\n",
    "    if len(arr) > 1:\n",
    "        mid = len(arr) // 2\n",
    "        left_half = deepcopy(arr[:mid])\n",
    "        right_half = deepcopy(arr[mid:])\n",
    "\n",
    "        left_feature = merge_sort(left_half)\n",
    "        right_feature = merge_sort(right_half)\n",
    "\n",
    "        # print(f'L & R: {left_feature}, {right_feature}')\n",
    "        feature = [] + left_feature + right_feature\n",
    "\n",
    "        i = j = k = 0\n",
    "\n",
    "        # Merge the two halves into the original list\n",
    "        while i < len(left_half) and j < len(right_half):\n",
    "            if left_half[i] < right_half[j]:\n",
    "                arr[k] = left_half[i]\n",
    "                i += 1\n",
    "                feature.append(-1)\n",
    "            else:\n",
    "                arr[k] = right_half[j]\n",
    "                j += 1\n",
    "                feature.append(1)\n",
    "            k += 1\n",
    "\n",
    "        while i < len(left_half):\n",
    "            arr[k] = left_half[i]\n",
    "            i += 1\n",
    "            k += 1\n",
    "            feature.append(1)\n",
    "\n",
    "        while j < len(right_half):\n",
    "            arr[k] = right_half[j]\n",
    "            j += 1\n",
    "            k += 1\n",
    "            feature.append(-1)\n",
    "\n",
    "        # print('Return: ', feature)\n",
    "        return feature[:-1]\n",
    "\n",
    "def featurize(x):\n",
    "    \"\"\"\n",
    "    Featurize the permutation vector into continuous space using merge kernel. Only available to permutation vector.\n",
    "    \"\"\"\n",
    "    assert len(x.shape) == 2, \"Only featurize 2 dimension permutation vector\"\n",
    "    feature = []\n",
    "    x_copy = deepcopy(x)\n",
    "    for arr in x_copy:\n",
    "        feature.append(merge_sort(arr))\n",
    "    normalizer = np.sqrt(x.size(1)*(x.size(1) - 1)/2)\n",
    "    return torch.tensor(feature/normalizer)\n",
    "\n",
    "\n",
    "a = [i for i in range(280)]\n",
    "a = torch.tensor([a])\n",
    "featurize(a).shape"
   ]
  }
 ],
 "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
}
