{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "from matplotlib import rcParams\n",
    "import pickle\n",
    "import jax\n",
    "from tabulate import tabulate\n",
    "import jax.numpy as jnp\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "jax.config.update(\"jax_enable_x64\", True)\n",
    "\n",
    "params = {\n",
    "    'axes.labelsize': 8,\n",
    "    'font.size': 8,\n",
    "    'legend.fontsize': 10,\n",
    "    'xtick.labelsize': 10,\n",
    "    'ytick.labelsize': 10,\n",
    "    'text.usetex': False,\n",
    "    'figure.figsize': [7.2, 2.2] #[4.5, 4.5]\n",
    "}\n",
    "rcParams.update(params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "avg_err_sm_list = []\n",
    "avg_err_aml_list = []\n",
    "avg_err_det_list = []\n",
    "avg_err_nn_list = []\n",
    "\n",
    "std_err_sm_list = []\n",
    "std_err_aml_list = []\n",
    "std_err_det_list = []\n",
    "std_err_nn_list = []\n",
    "\n",
    "for i in range(10):\n",
    "    with open(f'../results/3_mnist_sm_obs_idx_{i}.pkl', 'rb') as f:\n",
    "        res = pickle.load(f)\n",
    "        err_list = jnp.array(res['err_list'])\n",
    "        avg_err_sm_list.append(jnp.mean(err_list))\n",
    "        std_err_sm_list.append(jnp.std(err_list))\n",
    "        \n",
    "    with open(f'../results/3_mnist_aml_obs_idx_{i}.pkl', 'rb') as f:\n",
    "        res = pickle.load(f)\n",
    "        err_list = jnp.array(res['err_list'])\n",
    "        avg_err_aml_list.append(jnp.mean(err_list))\n",
    "        std_err_aml_list.append(jnp.std(err_list))\n",
    "        \n",
    "    with open(f'../results/3_mnist_det_obs_idx_{i}.pkl', 'rb') as f:\n",
    "        res = pickle.load(f)\n",
    "        err_list = jnp.array(res['err_list'])\n",
    "        avg_err_det_list.append(jnp.mean(err_list))\n",
    "        std_err_det_list.append(jnp.std(err_list))\n",
    "\n",
    "    with open(f'./3_mnist_sm_NN_obs_idx_{i}.pkl', 'rb') as f:\n",
    "        res = pickle.load(f)\n",
    "        err_list = jnp.array(res['err_list'])\n",
    "        avg_err_nn_list.append(jnp.mean(err_list))\n",
    "        std_err_nn_list.append(jnp.std(err_list))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[Array(0.16351594, dtype=float32),\n",
       " Array(0.08812707, dtype=float32),\n",
       " Array(0.1997751, dtype=float32),\n",
       " Array(0.06343082, dtype=float32),\n",
       " Array(0.12512033, dtype=float32),\n",
       " Array(0.27744442, dtype=float32),\n",
       " Array(0.09523833, dtype=float32),\n",
       " Array(0.14053406, dtype=float32),\n",
       " Array(0.05882514, dtype=float32),\n",
       " Array(0.22171621, dtype=float32)]"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "std_err_nn_list"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "|              NN |              SM |             AML |             DET |\n",
      "|-----------------|-----------------|-----------------|-----------------|\n",
      "| 2.4963 (0.1635) | 2.0635 (0.0397) | 6.1476 (0.1594) | 1.6575 (0.0726) |\n",
      "| 2.2283 (0.1998) | 1.9612 (0.0533) | 6.2249 (0.0196) | 1.3495 (0.2286) |\n",
      "| 0.9394 (0.0634) | 0.7961 (0.0367) | 3.0341 (0.4033) | 0.6230 (0.0356) |\n",
      "| 0.8271 (0.0588) | 0.7863 (0.0427) | 1.5439 (0.1210) | 0.4170 (0.0142) |\n"
     ]
    }
   ],
   "source": [
    "rows = [\n",
    "    [       f\"{nn_mse:.4f} ({nn_std:.4f})\",\n",
    "            f\"{sm_mse:.4f} ({sm_std:.4f})\",\n",
    "            f\"{aml_mse:.4f} ({aml_std:.4f})\",\n",
    "            f\"{det_mse:.4f} ({det_std:.4f})\"]\n",
    "    for     nn_mse, nn_std,\n",
    "            sm_mse, sm_std,\n",
    "            aml_mse, aml_std,\n",
    "            det_mse, det_std in zip(avg_err_nn_list, std_err_nn_list, \n",
    "                                    avg_err_sm_list, std_err_sm_list,\n",
    "                                    avg_err_aml_list, std_err_aml_list,\n",
    "                                    avg_err_det_list, std_err_det_list)\n",
    "]\n",
    "\n",
    "table = tabulate(\n",
    "    rows,\n",
    "    headers=[\"NN\", \"SM\", \"AML\", \"DET\"],\n",
    "    tablefmt=\"github\",        \n",
    "    disable_numparse=[1],     \n",
    "    colalign=(\"right\", \"right\", \"right\", \"right\") \n",
    ")\n",
    "\n",
    "print(table)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "      NN        SM      AML       DET\n",
      "--------  --------  -------  --------\n",
      "2.49625   2.06345   6.1476   1.65752\n",
      "2.22834   1.96124   6.22488  1.34949\n",
      "0.939392  0.79607   3.03412  0.622968\n",
      "0.827145  0.786345  1.54391  0.417049\n"
     ]
    }
   ],
   "source": [
    "print(tabulate({\n",
    "    'NN': avg_err_nn_list,\n",
    "    'SM': avg_err_sm_list,\n",
    "    'AML': avg_err_aml_list,\n",
    "    'DET': avg_err_det_list\n",
    "}, headers=\"keys\"))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "       NN         SM        AML        DET\n",
      "---------  ---------  ---------  ---------\n",
      "0.163516   0.0397195  0.159438   0.0726238\n",
      "0.199775   0.0533136  0.0196123  0.228633\n",
      "0.0634308  0.0366754  0.403275   0.0356495\n",
      "0.0588251  0.0426773  0.121001   0.0142115\n"
     ]
    }
   ],
   "source": [
    "print(tabulate({\n",
    "    'NN': std_err_nn_list,\n",
    "    'SM': std_err_sm_list,\n",
    "    'AML': std_err_aml_list,\n",
    "    'DET': std_err_det_list\n",
    "}, headers=\"keys\"))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "df_list = [pd.concat([\n",
    "        pd.DataFrame({'Method': 'FSM', 'MSE': jnp.array(err_sm_list[i])}),\n",
    "        pd.DataFrame({'Method': 'Opt', 'MSE': jnp.array(err_det_list[i])}),\n",
    "        pd.DataFrame({'Method': 'KDE-SP', 'MSE': jnp.array(err_aml_list[i])}),\n",
    "    ], ignore_index=True) for i in range(10)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_394359/3520690567.py:30: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df_list[0],\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 325x325 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "params = {\n",
    "    'figure.figsize'  : [6.5, 3.25],\n",
    "    'axes.labelsize'  : 8,\n",
    "    'font.size'       : 8,\n",
    "    'legend.fontsize' : 8,\n",
    "    'xtick.labelsize' : 8,\n",
    "    'ytick.labelsize' : 8,\n",
    "    'text.usetex'     : False\n",
    "}\n",
    "rcParams.update(params)\n",
    "\n",
    "method_palette = {\n",
    "    'FSM'   : '#B22400',   \n",
    "    'KDE-SP': '#006BB2',\n",
    "    'Opt'   : '#009E73'}\n",
    "\n",
    "\n",
    "box_kws = dict(width=0.5,\n",
    "               linewidth=1.2,\n",
    "               medianprops  = {'color': 'black', 'linewidth': 1.2},\n",
    "               whiskerprops = {'linewidth': 1.0},\n",
    "               capprops     = {'linewidth': 1.0},\n",
    "               fliersize    = 2)\n",
    "\n",
    "fig_w, fig_h = rcParams['figure.figsize']  \n",
    "quarter_width = fig_w / 2                       \n",
    "\n",
    "fig, ax = plt.subplots(figsize=(quarter_width, fig_h))\n",
    "\n",
    "sns.boxplot(data=df_list[0],         \n",
    "            x='Method', y='MSE',\n",
    "            ax=ax,\n",
    "            palette=method_palette,\n",
    "            **box_kws)\n",
    "\n",
    "ax.set_xlabel('')\n",
    "ax.set_ylabel(r'Prediction Error: $\\|\\hat{X}-X_0\\|$')\n",
    "ax.grid(axis='y', alpha=0.3)\n",
    "\n",
    "\n",
    "plt.tight_layout()\n",
    "fig.subplots_adjust(bottom=0.18)\n",
    "plt.savefig('mnist_single.pdf', bbox_inches='tight')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n",
      "/tmp/ipykernel_394359/2962464127.py:15: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df, x='Method', y='MSE',\n"
     ]
    },
    {
     "data": {
      "image/png": "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zPv44481YsQIDRo0SJKUnJysc845p8uxkydPVmpqqkKhUI8/j/6OjHaNjJJRpyCjXSOjZNQpyGjXyCgZdQoy2jUySkZpdsXg2muv7VjSFwwGdfvtt2vs2LEqKirSAw88oPfee6/T+CFDhujb3/62XnzxRdXU1MgwDI0fP96m6vvGtm3bNHbsWJ188smd3n/hhRfqk08+UX19vXbs2KHp06dr69atmjNnjn7wgx9Ikh5//HFNmDBB27dv13/+53/aUb4lrrvuOhUWFuqhhx6KOC43N1d79+5VfX29JOnAgQOdlo3edNNNXX5ccXGxdu/erdNPP1033XST/uM//kNNTU1djv3Tn/6klpYWjR07Nr5Pqp8go9GRUTJqJzIaHRklo3Yio9GRUTJqJzIaHRklo8l99kwu8vzzz+uss87qOJ42bZp++tOfav369fqf//kfTZ48Wa+99pouuuiijjEzZ87U/fff39E9TgQejyfi41//+tdVUFAgSfr+97+vW2+9VZ9++mkfVNY3/umf/kmrV6/W7bffLr/f3+24r3bQ09PTtX379qjzf+1rX9Pbb7+tLVu2aPPmzfrtb3+r3/zmN9qyZYsGDhwoScrLy5PX69WwYcP00ksv6aSTTorrc+ovyKg5ZJSM2oWMmkNGyahdyKg5ZJSM2oWMmkNGEzujNLssctppp2nGjBmaMWOGhgwZoj/84Q+dvrnk5eVpz549qq6u1rvvvqutW7faWG3vmzx5st577z3t27evUzf9r3/9qzIzM5WRkdHlx0X7htSf/OxnP9OZZ56pgoICrV+/XmPGjOlyXGVlpUaOHKmMjAx99NFH3c531113aePGjZKklStX6uyzz1ZSUpLy8vKUl5enu+66S6eccop27typc889V5L0+uuvd1rCnMjIaGdklIw6DRntjIySUacho52RUTLqNGS0MzJKRml2xenQoUP6y1/+oiuuuEIej0dNTU2qrq7WNddcc8LYX//61/r88881dOhQGyrtW2PHjtXUqVN16623auXKlRo8eLBCoZDuuece/fznP5d07DrqjRs3Kj8/X88//7xGjx6tUaNG6e9//7sOHjxo82dgjTlz5sjr9XZ8g/mqHTt2aPbs2br33nujfmP99a9/3em4qqpKw4YN0xlnnCFJqqmp0dGjRyN27RMRGe0aGT2GjNqPjHaNjB5DRu1H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mtgZZWVXnBobG9XY2KhRo0Z1vC87O1upqakdx9ddd51efvlltbS0SJJWrFihmTNn9nmtfen999/Xiy++qCeffFJDhgyRdKy7vGTJEj3wwAMKBoM655xzdPPNN+u8885TTk6O3nrrLUnSbbfdpnfffVeTJk3S1VdfbeenEZf6+npdcskluuCCC/T444/L6z0xbhMnTtTSpUu1aNEiHft+YN7HH3+sESNGaNCgQZKk5ORknXPOOV2OnTx5slJTUxUKhXr8efR3ZLRrZJSMOgUZ7RoZJaNOQUa7RkbJqFOQ0a6RUTJKsysG1157bceSvmAwqNtvv11jx45VUVGRHnjgAb333nudxg8ZMkTf/va39eKLL6qmpkaGYWj8+PE2Vd83tm3bprFjx+rkk0/u9P4LL7xQn3zyierr67Vjxw5Nnz5dW7du1Zw5c/SDH/xAkvT4449rwoQJ2r59u/7zP//TjvItcd1116mwsFAPPfRQxHG5ubnau3ev6uvrJUkHDhzotGz0pptu6vLjiouLtXv3bp1++um66aab9B//8R9qamrqcuyf/vQntbS0aOzYsfF9Uv0EGY2OjJJRO5HR6MgoGbUTGY2OjJJRO5HR6MgoGU3us2dykeeff15nnXVWx/G0adP005/+VOvXr9f//M//aPLkyXrttdd00UUXdYyZOXOm7r///o7ucSLweDwRH//617+ugoICSdL3v/993Xrrrfr000/7oLK+8U//9E9avXq1br/9dvn9/m7HfbWDnp6eru3bt0ed/2tf+5refvttbdmyRZs3b9Zvf/tb/eY3v9GWLVs0cOBASVJeXp68Xq+GDRuml156SSeddFJcn1N/QUbNIaNk1C5k1BwySkbtQkbNIaNk1C5k1BwymtgZpdllkdNOO00zZszQjBkzNGTIEP3hD3/o9M0lLy9Pe/bsUXV1td59911t3brVxmp73+TJk/Xee+9p3759nbrpf/3rX5WZmamMjIwuPy7aN6T+5Gc/+5nOPPNMFRQUaP369RozZkyX4yorKzVy5EhlZGToo48+6na+u+66Sxs3bpQkrVy5UmeffbaSkpKUl5envLw83XXXXTrllFO0c+dOnXvuuZKk119/vdMS5kRGRjsjo2TUachoZ2SUjDoNGe2MjJJRpyGjnZFRMkqzK06HDh3SX/7yF11xxRXyeDxqampSdXW1rrnmmhPG/vrXv9bnn3+uoUOH2lBp3xo7dqymTp2qW2+9VStXrtTgwYMVCoV0zz336Oc//7mkY9dRb9y4Ufn5+Xr++ec1evRojRo1Sn//+9918OBBmz8Da8yZM0der7fjG8xX7dixQ7Nnz9a9994b9Rvrr3/9607HVVVVGjZsmM444wxJUk1NjY4ePRqxa5+IyGjXyOgxZNR+ZLRrZPQYMmo/Mto1MnoMGbUfGe0aGT0mkTNKsytOhmHo8ccf16xZs+Tz+XT06FFdccUV+slPfnLC2Msuu8yGCu3zu9/9Tv/6r/+qs88+WwMHDlRSUpLuuece3XLLLQoGg5o0aZJWrVqlu+++W4Zh6LnnnpN0bJO87OxsnXXWWTr99NP79XXSklRSUiKv16uLL75YF198sR566CEtX75chw8f1siRIzV37txO10G3XyN9vL/+9a/y+Xyd3rdv3z7dcccdOnDggHw+n5KSkvTcc891+ypFoiKj3SOjx5BRe5HR7pHRY8iovcho98joMWTUXmS0e2T0mETNqMfo6Zb7gAWCwaBKSkpUVVVldykAukBGAWcjo4CzkVHA2cio+3E3RgAAAAAAALgGK7sAAAAAAADgGqzsAgAAAAAAgGvQ7AIAAAAAAIBr0OwCAAAAAACAa9DsAgAAAAAAgGvQ7AIAAAAAAIBr0OwCAAAAAACAa9DsAgAAAAAAgGvQ7AIAAAAAAIBr/H9+P+37Xak7bAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1200x500 with 10 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig_w, fig_h = 12, 5                     \n",
    "n_rows, n_cols = 2, 5\n",
    "\n",
    "fig, axes = plt.subplots(n_rows, n_cols, sharey=True,\n",
    "                         figsize=(fig_w, fig_h),\n",
    "                         layout='constrained')\n",
    "\n",
    "axes = axes.ravel()\n",
    "\n",
    "\n",
    "for i, (df, ax) in enumerate(zip(df_list, axes)):\n",
    "    sns.boxplot(data=df, x='Method', y='MSE',\n",
    "                ax=ax,\n",
    "                palette=method_palette,\n",
    "                **box_kws)\n",
    "\n",
    "    ax.set_xlabel('')\n",
    "    if i % n_cols == 0:                    \n",
    "        ax.set_ylabel(r'Pred. Error: $\\|\\hat{X}-X_0\\|$')\n",
    "    else:                                  \n",
    "        ax.set_ylabel('')\n",
    "\n",
    "    ax.tick_params(axis='y', which='major', labelsize=7)\n",
    "    ax.grid(axis='y', alpha=0.3)\n",
    "\n",
    "for ax in axes[len(df_list):]:\n",
    "    ax.set_visible(False)\n",
    "\n",
    "plt.savefig('mnist_full.pdf', bbox_inches='tight')\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "sm_approxml",
   "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.13.1"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
