{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "import os\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "NUM_SEEDS = 20\n",
    "DIMS = [2, 5, 10, 20 ,30]\n",
    "DIMS = [5, 10, 20 ,30,50]\n",
    "DIMS = [10, 20 ,30,50]\n",
    "\n",
    "AUTO_SCALING = True\n",
    "\n",
    "\n",
    "METHODS = ['les_250_8',  'mes','logei','loghvarei', 'turbo', 'hci_gibo_09','sobol'] #'grad', 'nograd', 'gibo',\n",
    "LABEL_NAMES = {'les_250_8':'LES (ours)',\n",
    "'mes':'MES',\n",
    "'turbo':'TuRBO',\n",
    "'sobol':'Sobol random',\n",
    "'logei':'logEI', \n",
    "'loghvarei':'logEI-DSP',\n",
    "'hci_gibo':'HCI-GIBO',\n",
    "'hci_gibo_09':'HCI-GIBO'}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [],
   "source": [
    "def decompress_gibo(df):\n",
    "    data = df[['y','n']].to_numpy(dtype=float)\n",
    "    repeats = np.diff(data[:,-1].astype(int))\n",
    "    repeats = np.insert(repeats, 0, data[0,-1])\n",
    "    repeated = np.repeat(data[:, :-1], repeats, axis=0)\n",
    "    return np.minimum.accumulate(repeated)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Comparison of performance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from matplotlib.lines import Line2D\n",
    "\n",
    "experiment_data = {}\n",
    "\n",
    "\n",
    "\n",
    "within_mdl_experiments =  ['within_mdl_high/gpsample','within_mdl_medium/gpsample','within_mdl_low/gpsample','within_mdl_ext_low/gpsample']\n",
    "oom_experiments = ['oom_high/gpsample','oom_medium/gpsample','oom_low/gpsample','oom_ext_low/gpsample']\n",
    "\n",
    "\n",
    "for within_mdl_experiment,oom_experiment,name in zip(within_mdl_experiments,oom_experiments,[\"high\",\"medium\",\"low\",\"ext_low\"]):\n",
    "\n",
    "\n",
    "    # load data\n",
    "\n",
    "    for experiment in [oom_experiment,within_mdl_experiment]:\n",
    "\n",
    "   \n",
    "        BEST_HIST_PATH = f\"./Data/{experiment}/optimizer_history/list_of_bests_\"\n",
    "        avrg_best_history = []\n",
    "        std_best_history = []\n",
    "        lower_quantiles_history = []\n",
    "        upper_quantiles_history = []\n",
    "\n",
    "        for dim in DIMS: \n",
    "            data_mean = []\n",
    "            stds = []\n",
    "            lower_quantile = []\n",
    "            upper_quantile = []\n",
    "            data_mean_normalized = []\n",
    "            stds_normalized = []\n",
    "            num_opjective_calls = min(20*dim, 400)\n",
    "            for method in range(len(METHODS)):\n",
    "                y_data = np.zeros((0,0))\n",
    "                for seed in range(NUM_SEEDS):\n",
    "                    file_identifier = f'{(seed+1):05d}_{dim}_{METHODS[method]}.csv'\n",
    "                    try: \n",
    "                        table = pd.read_csv(BEST_HIST_PATH + file_identifier) \n",
    "                    except: \n",
    "                        print(f'Unable to find file {BEST_HIST_PATH+file_identifier}.')\n",
    "                        continue\n",
    "                    \n",
    "                    if 'gibo' in METHODS[method]:\n",
    "                        new_data = decompress_gibo(table.dropna())\n",
    "                    else:\n",
    "                        new_data = np.reshape(table['y'].to_numpy(), [-1,1])\n",
    "                    if y_data.shape[0] == 0:\n",
    "                        y_data = new_data[:num_opjective_calls, :]\n",
    "                    else: \n",
    "                        new_data = new_data[:num_opjective_calls, :]\n",
    "                        y_data = np.concatenate([y_data, new_data], axis=1)\n",
    "                #data_mean.append(np.mean(y_data, axis=1))\n",
    "                data_mean.append(np.median(y_data, axis=1))\n",
    "                stds.append(np.std(y_data, axis=1))\n",
    "                \n",
    "                try:\n",
    "                    lower_quantile.append(np.quantile(y_data, 0.25, axis=1))\n",
    "                    upper_quantile.append(np.quantile(y_data, 0.75, axis=1))\n",
    "                except:\n",
    "                    lower_quantile.append([])\n",
    "                    upper_quantile.append([])\n",
    "\n",
    "            avrg_best_history.append(data_mean)\n",
    "            #avrg_best_history_normalized.append(data_mean_normalized)\n",
    "            std_best_history.append(stds)\n",
    "            #std_best_history_normalized.append(stds_normalized)\n",
    "            lower_quantiles_history.append(lower_quantile)\n",
    "            upper_quantiles_history.append(upper_quantile)\n",
    "        experiment_data[experiment] = {'avrg': avrg_best_history,\n",
    "                                    'std' : std_best_history,\n",
    "                                    'lower' : lower_quantiles_history,\n",
    "                                    'upper' : upper_quantiles_history}\n",
    "        \n",
    "    \n",
    "    import matplotlib as mpl\n",
    "    from matplotlib.ticker import MaxNLocator\n",
    "    from matplotlib.ticker import MultipleLocator\n",
    "\n",
    "    # Change font to Computer Modern\n",
    "    mpl.rcParams.update({\n",
    "        \"text.usetex\": True,\n",
    "        \"font.family\": \"serif\",  # LaTeX default is Computer Modern\n",
    "        \"font.serif\": [\"Computer Modern Roman\"],\n",
    "        \"axes.unicode_minus\": False  # to handle minus signs correctly\n",
    "    })\n",
    "\n",
    "\n",
    "    \n",
    "    colors = anonymized\n",
    "\n",
    "\n",
    "    width = 397.48 /72.27 # pt to in\n",
    "    aspect_ratio = 0.45\n",
    "    fontsize = 8\n",
    "    linewidth = 0.8\n",
    "    fig, axs = plt.subplots(2,len(DIMS), figsize=(width,width*aspect_ratio), sharey=True)\n",
    "\n",
    "    #############################\n",
    "    ### WITHIN MODEL          ###\n",
    "    #############################\n",
    "    avrg_best_history = experiment_data[within_mdl_experiment]['avrg']\n",
    "    upper_quantiles_history = experiment_data[within_mdl_experiment]['upper']\n",
    "    lower_quantiles_history = experiment_data[within_mdl_experiment]['lower']\n",
    "    for dim, ax in enumerate(axs.flat[:len(DIMS)]):\n",
    "        for method in range(len(METHODS)):\n",
    "            try:\n",
    "                ax.plot(avrg_best_history[dim][method], color=colors[method], linestyle='-', linewidth=linewidth, zorder=len(METHODS)-method+1)\n",
    "                x = np.arange(1, std_best_history[dim][method].shape[0]+1)\n",
    "                # ax.fill_between(x, avrg_best_history[dim][method] + std_best_history[dim][method], avrg_best_history[dim][method] - std_best_history[dim][method], \n",
    "                #                     color=colors[method], alpha=0.1)\n",
    "                ax.fill_between(x, upper_quantiles_history[dim][method], lower_quantiles_history[dim][method], \n",
    "                                color=colors[method], alpha=0.2)\n",
    "            except:\n",
    "                print(f\"Method number {method} not available\")\n",
    "        \n",
    "        ax.set_xlabel('')\n",
    "        ax.set_title(fr'$d = {DIMS[dim]}$', fontsize=fontsize)\n",
    "        ax.set_xlim(0, len(avrg_best_history[dim][0]))\n",
    "        ax.set_ylim(-11,0.6) \n",
    "        #ax.xaxis.set_major_locator(MaxNLocator(integer=True, nbins=2))  # Adjust number of x-axis gridlines\n",
    "        ax.xaxis.set_minor_locator(MaxNLocator(integer=True, nbins=4))  # Adjust number of y-axis gridlines\n",
    "        ax.yaxis.set_minor_locator(MaxNLocator(integer=True, nbins=10))\n",
    "        #ax.xaxis.set_minor_locator(MultipleLocator(50))  # or any consistent interval\n",
    "        ax.set_xticks([0,np.ceil(len(avrg_best_history[dim][0])/100)*50,np.ceil(len(avrg_best_history[dim][0])/100)*100])\n",
    "        ax.tick_params(axis='x', labelbottom=False)\n",
    "\n",
    "        #ax.minorticks_on()\n",
    "        ax.tick_params(axis='both', which='both', size=0)\n",
    "        ax.grid(which='major', linestyle='-', linewidth=0.5, alpha=0.7) \n",
    "        ax.grid(which='minor', linestyle='--', linewidth=0.3, alpha=0.7) \n",
    "        ax.tick_params(axis='both', which='major', labelsize=fontsize)\n",
    "\n",
    "        for spine in ax.spines.values():\n",
    "            spine.set_zorder(100)\n",
    "\n",
    "    #############################\n",
    "    ### OUT OF MODEL          ###\n",
    "    #############################\n",
    "    avrg_best_history = experiment_data[oom_experiment]['avrg']\n",
    "    upper_quantiles_history = experiment_data[oom_experiment]['upper']\n",
    "    lower_quantiles_history = experiment_data[oom_experiment]['lower']\n",
    "    for dim, ax in enumerate(axs.flat[len(DIMS):]):\n",
    "        for method in range(len(METHODS)):\n",
    "            ax.plot(avrg_best_history[dim][method], color=colors[method], linestyle='-', linewidth=linewidth, zorder=len(METHODS)-method+1)\n",
    "            x = np.arange(1, upper_quantiles_history[dim][method].shape[0]+1)\n",
    "            # ax.fill_between(x, avrg_best_history[dim][method] + std_best_history[dim][method], avrg_best_history[dim][method] - std_best_history[dim][method], \n",
    "            #                     color=colors[method], alpha=0.1)\n",
    "            ax.fill_between(x, upper_quantiles_history[dim][method], lower_quantiles_history[dim][method], \n",
    "                                color=colors[method], alpha=0.2)\n",
    "        \n",
    "        ax.set_xlabel('')\n",
    "        #ax.set_title(fr'$d = {DIMS[dim]}$', fontsize=fontsize)\n",
    "        ax.set_xlim(0, len(avrg_best_history[dim][0]))\n",
    "        ax.set_ylim(-11,0.6) \n",
    "        #ax.xaxis.set_major_locator(MaxNLocator(integer=True, nbins=2))  # Adjust number of x-axis gridlines\n",
    "        ax.xaxis.set_minor_locator(MaxNLocator(integer=True, nbins=4))  # Adjust number of y-axis gridlines\n",
    "        ax.yaxis.set_minor_locator(MaxNLocator(integer=True, nbins=10))\n",
    "        #ax.xaxis.set_minor_locator(MultipleLocator(50))  # or any consistent interval\n",
    "        ax.set_xticks([0,np.ceil(len(avrg_best_history[dim][0])/100)*50,np.ceil(len(avrg_best_history[dim][0])/100)*100])\n",
    "\n",
    "        #ax.minorticks_on()\n",
    "        ax.tick_params(axis='both', which='both', size=0)\n",
    "        ax.grid(which='major', linestyle='-', linewidth=0.5, alpha=0.7) \n",
    "        ax.grid(which='minor', linestyle='--', linewidth=0.3, alpha=0.7) \n",
    "        ax.tick_params(axis='both', which='major', labelsize=fontsize)\n",
    "\n",
    "        for spine in ax.spines.values():\n",
    "            spine.set_zorder(100)\n",
    "\n",
    "\n",
    "    #############################\n",
    "    ### GENERAL               ###\n",
    "    #############################\n",
    "\n",
    "    axs[0][0].set_yticks([x for x in range(-10,1,2)])\n",
    "    #axs.flat[0].set_yticks(np.linspace(0, -2, 3))\n",
    "    axs.flat[0].set_ylabel(r'$f(\\hat{\\mathbf{x}}^*)$', fontsize=fontsize)\n",
    "    axs.flat[len(DIMS)].set_ylabel(r'$f(\\hat{\\mathbf{x}}^*)$', fontsize=fontsize)\n",
    "    fig.text(0.5, 0.01, r'Evaluations \\#', ha='center', fontsize=fontsize)\n",
    "    legend_elements = [Line2D([0], [0], color=colors[i], lw=linewidth*1.2, linestyle='-', label=LABEL_NAMES[METHODS[i]]) for i in range(len(METHODS))]\n",
    "\n",
    "    fig.text(0.5, 0.92, \"Within Model Comparison\", ha='center', va='center', fontsize=fontsize+1)\n",
    "    fig.text(0.5, 0.45, \"Out of Model Comparison\", ha='center', va='center', fontsize=fontsize+1)\n",
    "\n",
    "    fig.tight_layout(rect=[0, 0, 1, 0.93], h_pad=2.1)  # adjust layout to leave space below\n",
    "    fig.legend(\n",
    "        handles=legend_elements,\n",
    "        loc='lower center',\n",
    "        bbox_to_anchor=(0.5, -0.2),#bbox_to_anchor=(0.5, -0.1),\n",
    "        ncol=4,#len(METHODS),\n",
    "        fontsize=fontsize,\n",
    "        frameon=False,\n",
    "    )\n",
    "    plt.subplots_adjust(wspace=0.2)\n",
    "    fig.savefig(name + \"_both_comparisons.pdf\", bbox_inches='tight', pad_inches=0.0)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "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.12.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
