{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "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": [
    "\n",
    "NUM_SEEDS = 20\n",
    "DIMS = [5, 10, 20 ,30,50]\n",
    "\n",
    "\n",
    "function = 'ablations_low/gpsample'\n",
    "function_name = 'low'\n",
    "function = 'ablations_medium/gpsample'\n",
    "function_name = 'medium'\n",
    "function = 'ablations_high/gpsample'\n",
    "function_name = 'high'\n",
    "\n",
    "\n",
    "\n",
    "DISTR_PATH = \"./Data/\"+function+\"/local_optima_distribution/\"\n",
    "BEST_HIST_PATH = \"./Data/\"+function+\"/optimizer_history/list_of_bests_\"\n",
    "SAMPLED_POINTS_PATH = \"./Data/\"+function+\"/sampled_data/sampled_data_history_\"\n",
    "LENGTH_SCALE_PATH = \"./Data/\"+function+\"/length_scale/length_scale_\"\n",
    "\n",
    "\n",
    "           \n",
    "ABLATIONS = [['les_250_8','les_250_4','les_250_16','les_20_8','lesgradcond_20_8'],['les_250_8','les_beta05_250_8','lesGD_250_8','les_20_8','lesCMAES_20_8'],['les_250_8','les_fp_wgrad','localTS']]\n",
    "ABLATION_NAMES = ['apdx_discrtization_ablation','apdx_difopt_ablations','apdx_other_info_methods']\n",
    "\n",
    "counter = -1\n",
    "for METHODS in ABLATIONS: \n",
    "    counter += 1\n",
    "\n",
    "    LABEL_NAMES = {'tracing':'With gradient tracing',\n",
    "    'mes':'MES',\n",
    "    'logei':'logEI',\n",
    "    'turbo':'TURBO',\n",
    "    'sobol':'Sobol random',\n",
    "    'std_gibo':'GIBO',\n",
    "    'hci_gibo':'HCI-GIBO',\n",
    "    'hci_gibo_09':'HCI-GIBO',\n",
    "    'les_20_8':'LES-ADAM: L = 20, P = 8',\n",
    "    'les_250_4':'LES-ADAM: L = 250, P = 4',\n",
    "    'les_250_16':'LES-ADAM: L = 250, P = 16',\n",
    "    'les_250_8':'LES-ADAM: L = 250, P = 8 (default)',\n",
    "    'les_beta05_250_8':'LES-ADAM (beta_1 = 0.5): L = 250, P = 8',\n",
    "    'les_fp_wgrad':'LES-ADAM-Opt.Cond.: L = 250, P = 8 ',\n",
    "    'lesGD_250_8':'LES-GD: L = 250, P = 8',\n",
    "    'lesgradcond_20_8':'LES-ADAM-Grad.-Cond.: L = 20, P = 8',\n",
    "    'localTS':'Local Thompson Sampling',\n",
    "    'lesCMAES_20_8':'LES-CMAES: L = 20, P = 8'}\n",
    "    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)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    from matplotlib.lines import Line2D\n",
    "\n",
    "    avrg_best_history = []\n",
    "    std_best_history = []\n",
    "    lower_quantiles_history =[]\n",
    "    upper_quantiles_history =[]\n",
    "    avrg_time_deltas = []\n",
    "    std_time_deltas = []\n",
    "\n",
    "\n",
    "    for dim in DIMS: \n",
    "        data_mean = []\n",
    "        stds = []\n",
    "\n",
    "        time_deltas_mean_per_method = []\n",
    "        time_deltas_std_per_method = []\n",
    "        lower_quantile =[]\n",
    "        upper_quantile =[]\n",
    "\n",
    "\n",
    "        num_objective_calls = min(20*dim, 400)\n",
    "        for method in range(len(METHODS)):\n",
    "            y_data = np.zeros((0,0))\n",
    "            # Collect timestamp differences for all seeds having a timestamp column\n",
    "            time_arrays = []\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 METHODS[method] == 'std_gibo' or METHODS[method] == 'hci_gibo' or METHODS[method] == 'hci_gibo_09' :\n",
    "                    new_data = decompress_gibo(table.dropna())\n",
    "                else:\n",
    "                    new_data = np.reshape(table['f'].to_numpy(), [-1,1])\n",
    "                if y_data.shape[0] == 0:\n",
    "                    y_data = new_data[:num_objective_calls, :]\n",
    "                else: \n",
    "                    new_data = new_data[:num_objective_calls, :]\n",
    "                    y_data = np.concatenate([y_data, new_data], axis=1)\n",
    "\n",
    "                if 'timestamp' in table.columns:\n",
    "                    # Crop timestamps to match the # of objective calls\n",
    "                    timestamps = table['timestamp'].to_numpy()[:num_objective_calls]\n",
    "                    # Compute the consecutive differences\n",
    "                    if len(timestamps) > 1:\n",
    "                        diffs = np.diff(timestamps)\n",
    "                        time_arrays.append(diffs)\n",
    "\n",
    "\n",
    "                \n",
    "            data_mean.append(np.median(y_data, axis=1))\n",
    "            stds.append(np.std(y_data, axis=1))\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",
    "            stds.append(np.std(y_data, axis=1))\n",
    "\n",
    "            # Collect mean/std across seeds for the time-deltas (if available)\n",
    "            if len(time_arrays) > 0:\n",
    "                time_arrays = np.array(time_arrays)  # shape: [num_seeds_with_timestamps, num_objective_calls-1]\n",
    "                time_deltas_mean_per_method.append(np.mean(time_arrays, axis=0))\n",
    "                time_deltas_std_per_method.append(np.std(time_arrays, axis=0))\n",
    "            else:\n",
    "                # No timestamp data for this method in this dimension\n",
    "                time_deltas_mean_per_method.append(None)\n",
    "                time_deltas_std_per_method.append(None)\n",
    "\n",
    "        avrg_best_history.append(data_mean)\n",
    "        std_best_history.append(stds)\n",
    "        lower_quantiles_history.append(lower_quantile)\n",
    "        upper_quantiles_history.append(upper_quantile)\n",
    "\n",
    "        avrg_time_deltas.append(time_deltas_mean_per_method)\n",
    "        std_time_deltas.append(time_deltas_std_per_method)\n",
    "    pass\n",
    "   \n",
    "    colors = anonymized\n",
    "    fig, axs = plt.subplots(2,3, figsize=(14*2/3,6))\n",
    "\n",
    "    for dim, ax in enumerate(axs.flat[:5]):\n",
    "        for method in range(len(METHODS)):\n",
    "            try:\n",
    "                ax.plot(avrg_best_history[dim][method], color=colors[method], linestyle='-')\n",
    "                ax.set_title(fr'$d = {DIMS[dim]}$', fontsize=10)\n",
    "                x = np.arange(1, lower_quantiles_history[dim][method].shape[0]+1)\n",
    "                \n",
    "                ax.fill_between(x,lower_quantiles_history[dim][method], upper_quantiles_history[dim][method], \n",
    "                                    color=colors[method], alpha=0.1)\n",
    "                ax.grid(True)\n",
    "                ax.set_xlabel('Objective function evaluation')\n",
    "                ax.set_xlim(0,min(20*DIMS[dim], 400))\n",
    "                #ax.set_xlim(0, len(avrg_best_history[dim][0]))\n",
    "            except:\n",
    "                print('error!!!')\n",
    "                pass\n",
    "\n",
    "    axs.flat[5].axis('off')\n",
    "    axs.flat[0].set_yticks(np.linspace(0, -2, 3))\n",
    "    axs.flat[0].set_ylabel(r'Current best $f$')\n",
    "    axs.flat[3].set_ylabel(r'Current best $f$')\n",
    "    #fig.suptitle(f'Optimization history')\n",
    "    fig.tight_layout()\n",
    "\n",
    "\n",
    "    legend_elements = [Line2D([0], [0], color=colors[i], lw=1.5, linestyle='-', label=LABEL_NAMES[METHODS[i]]) for i in range(len(METHODS))]\n",
    "\n",
    "    axs.flat[5].legend(handles=legend_elements, loc='center',frameon=False)\n",
    "\n",
    "    plt.savefig(\"plots/ablations/\" + ABLATION_NAMES[counter] + \"_\"+function_name+\".pdf\", format=\"pdf\", bbox_inches=\"tight\", pad_inches=0.0)\n"
   ]
  }
 ],
 "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.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
